* with this program; if not, write  to the Free Software Foundation, Inc.,
  * 675 Mass Ave, Cambridge, MA 02139, USA.
  */
-#include <linux/gpio.h>
+#include <linux/gpio/consumer.h>
+#include <linux/gpio/driver.h>
 #include <linux/gpio/machine.h>
 #include <linux/kernel.h>
 #include <linux/init.h>
 /* TPS65010 has four GPIOs.  nPG and LED2 can be treated like GPIOs with
  * alternate pin configurations for hardware-controlled blinking.
  */
-#define OSK_TPS_GPIO_BASE              (OMAP_MAX_GPIO_LINES + 16 /* MPUIO */)
-#      define OSK_TPS_GPIO_USB_PWR_EN  (OSK_TPS_GPIO_BASE + 0)
-#      define OSK_TPS_GPIO_LED_D3      (OSK_TPS_GPIO_BASE + 1)
-#      define OSK_TPS_GPIO_LAN_RESET   (OSK_TPS_GPIO_BASE + 2)
-#      define OSK_TPS_GPIO_DSP_PWR_EN  (OSK_TPS_GPIO_BASE + 3)
-#      define OSK_TPS_GPIO_LED_D9      (OSK_TPS_GPIO_BASE + 4)
-#      define OSK_TPS_GPIO_LED_D2      (OSK_TPS_GPIO_BASE + 5)
+#define OSK_TPS_GPIO_USB_PWR_EN        0
+#define OSK_TPS_GPIO_LED_D3    1
+#define OSK_TPS_GPIO_LAN_RESET 2
+#define OSK_TPS_GPIO_DSP_PWR_EN        3
+#define OSK_TPS_GPIO_LED_D9    4
+#define OSK_TPS_GPIO_LED_D2    5
 
 static struct mtd_partition osk_partitions[] = {
        /* bootloader (U-Boot, etc) in first sector */
        /* NOTE:  D9 and D2 have hardware blink support.
         * Also, D9 requires non-battery power.
         */
-       { .gpio = OSK_TPS_GPIO_LED_D9, .name = "d9",
-                       .default_trigger = "disk-activity", },
-       { .gpio = OSK_TPS_GPIO_LED_D2, .name = "d2", },
-       { .gpio = OSK_TPS_GPIO_LED_D3, .name = "d3", .active_low = 1,
-                       .default_trigger = "heartbeat", },
+       { .name = "d9", .default_trigger = "disk-activity", },
+       { .name = "d2", },
+       { .name = "d3", .default_trigger = "heartbeat", },
+};
+
+static struct gpiod_lookup_table tps_leds_gpio_table = {
+       .dev_id = "leds-gpio",
+       .table = {
+               /* Use local offsets on TPS65010 */
+               GPIO_LOOKUP_IDX("tps65010", OSK_TPS_GPIO_LED_D9, NULL, 0, GPIO_ACTIVE_HIGH),
+               GPIO_LOOKUP_IDX("tps65010", OSK_TPS_GPIO_LED_D2, NULL, 1, GPIO_ACTIVE_HIGH),
+               GPIO_LOOKUP_IDX("tps65010", OSK_TPS_GPIO_LED_D3, NULL, 2, GPIO_ACTIVE_LOW),
+               { }
+       },
 };
 
 static struct gpio_led_platform_data tps_leds_data = {
        .dev.platform_data      = &tps_leds_data,
 };
 
-static int osk_tps_setup(struct i2c_client *client, void *context)
+/* The board just hold these GPIOs hogged from setup to teardown */
+static struct gpio_desc *eth_reset;
+static struct gpio_desc *vdd_dsp;
+
+static int osk_tps_setup(struct i2c_client *client, struct gpio_chip *gc)
 {
+       struct gpio_desc *d;
        if (!IS_BUILTIN(CONFIG_TPS65010))
                return -ENOSYS;
 
        /* Set GPIO 1 HIGH to disable VBUS power supply;
         * OHCI driver powers it up/down as needed.
         */
-       gpio_request(OSK_TPS_GPIO_USB_PWR_EN, "n_vbus_en");
-       gpio_direction_output(OSK_TPS_GPIO_USB_PWR_EN, 1);
+       d = gpiochip_request_own_desc(gc, OSK_TPS_GPIO_USB_PWR_EN, "n_vbus_en",
+                                     GPIO_ACTIVE_HIGH, GPIOD_OUT_HIGH);
        /* Free the GPIO again as the driver will request it */
-       gpio_free(OSK_TPS_GPIO_USB_PWR_EN);
+       gpiochip_free_own_desc(d);
 
        /* Set GPIO 2 high so LED D3 is off by default */
        tps65010_set_gpio_out_value(GPIO2, HIGH);
 
        /* Set GPIO 3 low to take ethernet out of reset */
-       gpio_request(OSK_TPS_GPIO_LAN_RESET, "smc_reset");
-       gpio_direction_output(OSK_TPS_GPIO_LAN_RESET, 0);
+       eth_reset = gpiochip_request_own_desc(gc, OSK_TPS_GPIO_LAN_RESET, "smc_reset",
+                                             GPIO_ACTIVE_HIGH, GPIOD_OUT_LOW);
 
        /* GPIO4 is VDD_DSP */
-       gpio_request(OSK_TPS_GPIO_DSP_PWR_EN, "dsp_power");
-       gpio_direction_output(OSK_TPS_GPIO_DSP_PWR_EN, 1);
+       vdd_dsp = gpiochip_request_own_desc(gc, OSK_TPS_GPIO_DSP_PWR_EN, "dsp_power",
+                                           GPIO_ACTIVE_HIGH, GPIOD_OUT_HIGH);
        /* REVISIT if DSP support isn't configured, power it off ... */
 
        /* Let LED1 (D9) blink; leds-gpio may override it */
 
        /* register these three LEDs */
        osk5912_tps_leds.dev.parent = &client->dev;
+       gpiod_add_lookup_table(&tps_leds_gpio_table);
        platform_device_register(&osk5912_tps_leds);
 
        return 0;
 }
 
+static void osk_tps_teardown(struct i2c_client *client, struct gpio_chip *gc)
+{
+       gpiochip_free_own_desc(eth_reset);
+       gpiochip_free_own_desc(vdd_dsp);
+}
+
 static struct tps65010_board tps_board = {
-       .base           = OSK_TPS_GPIO_BASE,
        .outmask        = 0x0f,
        .setup          = osk_tps_setup,
+       .teardown       = osk_tps_teardown,
 };
 
 static struct i2c_board_info __initdata osk_i2c_board_info[] = {
 {
        u32 l;
 
-       if ((gpio_request(0, "smc_irq")) < 0) {
-               printk("Error requesting gpio 0 for smc91x irq\n");
-               return;
-       }
-
        /* Check EMIFS wait states to fix errors with SMC_GET_PKT_HDR */
        l = omap_readl(EMIFS_CCS(1));
        l |= 0x3;
        struct resource *res = &osk5912_cf_resources[1];
 
        omap_cfg_reg(M7_1610_GPIO62);
-       if ((gpio_request(62, "cf_irq")) < 0) {
-               printk("Error requesting gpio 62 for CF irq\n");
-               return;
-       }
 
        switch (seg) {
        /* NOTE: CS0 could be configured too ... */
                seg, omap_readl(EMIFS_CCS(seg)), omap_readl(EMIFS_ACS(seg)));
        omap_writel(0x0004a1b3, EMIFS_CCS(seg));        /* synch mode 4 etc */
        omap_writel(0x00000000, EMIFS_ACS(seg));        /* OE hold/setup */
-
-       /* the CF I/O IRQ is really active-low */
-       irq_set_irq_type(gpio_to_irq(62), IRQ_TYPE_EDGE_FALLING);
 }
 
 static struct gpiod_lookup_table osk_usb_gpio_table = {
        .dev_id = "ohci",
        .table = {
                /* Power GPIO on the I2C-attached TPS65010 */
-               GPIO_LOOKUP("tps65010", 0, "power", GPIO_ACTIVE_HIGH),
+               GPIO_LOOKUP("tps65010", OSK_TPS_GPIO_USB_PWR_EN, "power",
+                           GPIO_ACTIVE_HIGH),
                GPIO_LOOKUP(OMAP_GPIO_LABEL, 9, "overcurrent",
                            GPIO_ACTIVE_HIGH),
+               { }
        },
 };
 
 
 #define EMIFS_CS3_VAL  (0x88013141)
 
+static struct gpiod_lookup_table osk_irq_gpio_table = {
+       .dev_id = NULL,
+       .table = {
+               /* GPIO used for SMC91x IRQ */
+               GPIO_LOOKUP(OMAP_GPIO_LABEL, 0, "smc_irq",
+                           GPIO_ACTIVE_HIGH),
+               /* GPIO used for CF IRQ */
+               GPIO_LOOKUP("gpio-48-63", 14, "cf_irq",
+                           GPIO_ACTIVE_HIGH),
+               /* GPIO used by the TPS65010 chip */
+               GPIO_LOOKUP("mpuio", 1, "tps65010",
+                           GPIO_ACTIVE_HIGH),
+               { }
+       },
+};
+
 static void __init osk_init(void)
 {
+       struct gpio_desc *d;
        u32 l;
 
        osk_init_smc91x();
 
        osk_flash_resource.end = osk_flash_resource.start = omap_cs3_phys();
        osk_flash_resource.end += SZ_32M - 1;
-       osk5912_smc91x_resources[1].start = gpio_to_irq(0);
-       osk5912_smc91x_resources[1].end = gpio_to_irq(0);
-       osk5912_cf_resources[0].start = gpio_to_irq(62);
-       osk5912_cf_resources[0].end = gpio_to_irq(62);
+
+       /*
+        * Add the GPIOs to be used as IRQs and immediately look them up
+        * to be passed as an IRQ resource. This is ugly but should work
+        * until the day we convert to device tree.
+        */
+       gpiod_add_lookup_table(&osk_irq_gpio_table);
+
+       d = gpiod_get(NULL, "smc_irq", GPIOD_IN);
+       if (IS_ERR(d)) {
+               pr_err("Unable to get SMC IRQ GPIO descriptor\n");
+       } else {
+               irq_set_irq_type(gpiod_to_irq(d), IRQ_TYPE_EDGE_RISING);
+               osk5912_smc91x_resources[1] = DEFINE_RES_IRQ(gpiod_to_irq(d));
+       }
+
+       d = gpiod_get(NULL, "cf_irq", GPIOD_IN);
+       if (IS_ERR(d)) {
+               pr_err("Unable to get CF IRQ GPIO descriptor\n");
+       } else {
+               /* the CF I/O IRQ is really active-low */
+               irq_set_irq_type(gpiod_to_irq(d), IRQ_TYPE_EDGE_FALLING);
+               osk5912_cf_resources[0] = DEFINE_RES_IRQ(gpiod_to_irq(d));
+       }
+
        platform_add_devices(osk5912_devices, ARRAY_SIZE(osk5912_devices));
 
        l = omap_readl(USB_TRANSCEIVER_CTRL);
        gpiod_add_lookup_table(&osk_usb_gpio_table);
        omap1_usb_init(&osk_usb_config);
 
+       omap_serial_init();
+
        /* irq for tps65010 chip */
        /* bootloader effectively does:  omap_cfg_reg(U19_1610_MPUIO1); */
-       if (gpio_request(OMAP_MPUIO(1), "tps65010") == 0)
-               gpio_direction_input(OMAP_MPUIO(1));
-
-       omap_serial_init();
-       osk_i2c_board_info[0].irq = gpio_to_irq(OMAP_MPUIO(1));
+       d = gpiod_get(NULL, "tps65010", GPIOD_IN);
+       if (IS_ERR(d))
+               pr_err("Unable to get TPS65010 IRQ GPIO descriptor\n");
+       else
+               osk_i2c_board_info[0].irq = gpiod_to_irq(d);
        omap_register_i2c_bus(1, 400, osk_i2c_board_info,
                              ARRAY_SIZE(osk_i2c_board_info));
 }