Skip to content

Commit

Permalink
mmc: jz4740: Use GPIO descriptor for power
Browse files Browse the repository at this point in the history
The power GPIO line is passed with inversion flags and all from
the platform data. Switch to using an optional GPIO descriptor and
use this to switch the power.

Augment the only boardfile to pass in the proper "power" descriptor
in the GPIO descriptor machine table instead.

As the GPIO handling is now much simpler, we can cut down on some
overhead code.

Cc: Paul Cercueil <paul@crapouillou.net>
Cc: linux-mips@linux-mips.org
Signed-off-by: Linus Walleij <linus.walleij@linaro.org>
Acked-by: Paul Burton <paul.burton@mips.com>
Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
  • Loading branch information
Linus Walleij authored and Ulf Hansson committed Dec 17, 2018
1 parent 0c901c0 commit 0f6f323
Show file tree
Hide file tree
Showing 3 changed files with 18 additions and 55 deletions.
2 changes: 0 additions & 2 deletions arch/mips/include/asm/mach-jz4740/jz4740_mmc.h
Original file line number Diff line number Diff line change
Expand Up @@ -3,10 +3,8 @@
#define __LINUX_MMC_JZ4740_MMC

struct jz4740_mmc_platform_data {
int gpio_power;
unsigned card_detect_active_low:1;
unsigned read_only_active_low:1;
unsigned power_active_low:1;

unsigned data_1bit:1;
};
Expand Down
6 changes: 2 additions & 4 deletions arch/mips/jz4740/board-qi_lb60.c
Original file line number Diff line number Diff line change
Expand Up @@ -43,8 +43,6 @@
#include "clock.h"

/* GPIOs */
#define QI_LB60_GPIO_SD_VCC_EN_N JZ_GPIO_PORTD(2)

#define QI_LB60_GPIO_KEYOUT(x) (JZ_GPIO_PORTC(10) + (x))
#define QI_LB60_GPIO_KEYIN(x) (JZ_GPIO_PORTD(18) + (x))
#define QI_LB60_GPIO_KEYIN8 JZ_GPIO_PORTD(26)
Expand Down Expand Up @@ -385,14 +383,14 @@ static struct platform_device qi_lb60_gpio_keys = {
};

static struct jz4740_mmc_platform_data qi_lb60_mmc_pdata = {
.gpio_power = QI_LB60_GPIO_SD_VCC_EN_N,
.power_active_low = 1,
/* Intentionally left blank */
};

static struct gpiod_lookup_table qi_lb60_mmc_gpio_table = {
.dev_id = "jz4740-mmc.0",
.table = {
GPIO_LOOKUP("GPIOD", 0, "cd", GPIO_ACTIVE_HIGH),
GPIO_LOOKUP("GPIOD", 2, "power", GPIO_ACTIVE_LOW),
{ },
},
};
Expand Down
65 changes: 16 additions & 49 deletions drivers/mmc/host/jz4740_mmc.c
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@
#include <linux/dmaengine.h>
#include <linux/dma-mapping.h>
#include <linux/err.h>
#include <linux/gpio.h>
#include <linux/gpio/consumer.h>
#include <linux/interrupt.h>
#include <linux/io.h>
#include <linux/irq.h>
Expand Down Expand Up @@ -136,6 +136,7 @@ struct jz4740_mmc_host {
struct platform_device *pdev;
struct jz4740_mmc_platform_data *pdata;
struct clk *clk;
struct gpio_desc *power;

enum jz4740_mmc_version version;

Expand Down Expand Up @@ -903,18 +904,16 @@ static void jz4740_mmc_set_ios(struct mmc_host *mmc, struct mmc_ios *ios)
switch (ios->power_mode) {
case MMC_POWER_UP:
jz4740_mmc_reset(host);
if (host->pdata && gpio_is_valid(host->pdata->gpio_power))
gpio_set_value(host->pdata->gpio_power,
!host->pdata->power_active_low);
if (host->power)
gpiod_set_value(host->power, 1);
host->cmdat |= JZ_MMC_CMDAT_INIT;
clk_prepare_enable(host->clk);
break;
case MMC_POWER_ON:
break;
default:
if (host->pdata && gpio_is_valid(host->pdata->gpio_power))
gpio_set_value(host->pdata->gpio_power,
host->pdata->power_active_low);
if (host->power)
gpiod_set_value(host->power, 0);
clk_disable_unprepare(host->clk);
break;
}
Expand Down Expand Up @@ -947,30 +946,9 @@ static const struct mmc_host_ops jz4740_mmc_ops = {
.enable_sdio_irq = jz4740_mmc_enable_sdio_irq,
};

static int jz4740_mmc_request_gpio(struct device *dev, int gpio,
const char *name, bool output, int value)
{
int ret;

if (!gpio_is_valid(gpio))
return 0;

ret = gpio_request(gpio, name);
if (ret) {
dev_err(dev, "Failed to request %s gpio: %d\n", name, ret);
return ret;
}

if (output)
gpio_direction_output(gpio, value);
else
gpio_direction_input(gpio);

return 0;
}

static int jz4740_mmc_request_gpios(struct mmc_host *mmc,
struct platform_device *pdev)
static int jz4740_mmc_request_gpios(struct jz4740_mmc_host *host,
struct mmc_host *mmc,
struct platform_device *pdev)
{
struct jz4740_mmc_platform_data *pdata = dev_get_platdata(&pdev->dev);
int ret = 0;
Expand All @@ -995,19 +973,12 @@ static int jz4740_mmc_request_gpios(struct mmc_host *mmc,
if (ret == -EPROBE_DEFER)
return ret;

return jz4740_mmc_request_gpio(&pdev->dev, pdata->gpio_power,
"MMC read only", true, pdata->power_active_low);
}

static void jz4740_mmc_free_gpios(struct platform_device *pdev)
{
struct jz4740_mmc_platform_data *pdata = dev_get_platdata(&pdev->dev);

if (!pdata)
return;
host->power = devm_gpiod_get_optional(&pdev->dev, "power",
GPIOD_OUT_HIGH);
if (IS_ERR(host->power))
return PTR_ERR(host->power);

if (gpio_is_valid(pdata->gpio_power))
gpio_free(pdata->gpio_power);
return 0;
}

static const struct of_device_id jz4740_mmc_of_match[] = {
Expand Down Expand Up @@ -1053,7 +1024,7 @@ static int jz4740_mmc_probe(struct platform_device* pdev)
mmc->caps |= MMC_CAP_SDIO_IRQ;
if (!(pdata && pdata->data_1bit))
mmc->caps |= MMC_CAP_4_BIT_DATA;
ret = jz4740_mmc_request_gpios(mmc, pdev);
ret = jz4740_mmc_request_gpios(host, mmc, pdev);
if (ret)
goto err_free_host;
}
Expand Down Expand Up @@ -1104,7 +1075,7 @@ static int jz4740_mmc_probe(struct platform_device* pdev)
dev_name(&pdev->dev), host);
if (ret) {
dev_err(&pdev->dev, "Failed to request irq: %d\n", ret);
goto err_free_gpios;
goto err_free_host;
}

jz4740_mmc_clock_disable(host);
Expand Down Expand Up @@ -1135,8 +1106,6 @@ static int jz4740_mmc_probe(struct platform_device* pdev)
jz4740_mmc_release_dma_channels(host);
err_free_irq:
free_irq(host->irq, host);
err_free_gpios:
jz4740_mmc_free_gpios(pdev);
err_free_host:
mmc_free_host(mmc);

Expand All @@ -1155,8 +1124,6 @@ static int jz4740_mmc_remove(struct platform_device *pdev)

free_irq(host->irq, host);

jz4740_mmc_free_gpios(pdev);

if (host->use_dma)
jz4740_mmc_release_dma_channels(host);

Expand Down

0 comments on commit 0f6f323

Please sign in to comment.