This patch adds MMC multislot support for TI OMAP H4 2420 boards. Signed-off-by: David Cohen <david.cohen@xxxxxxxxxxx> Signed-off-by: Carlos Eduardo Aguiar <carlos.aguiar@xxxxxxxxxxx> Acked-by: Anderson Lizardo <anderson.lizardo@xxxxxxxxxxx> --- arch/arm/mach-omap2/Makefile | 2 +- arch/arm/mach-omap2/board-h4-mmc.c | 266 ++++++++++++++++++++++++++++++++++ arch/arm/mach-omap2/board-h4.c | 10 +- include/asm-arm/arch-omap/board-h4.h | 3 + 4 files changed, 277 insertions(+), 4 deletions(-) create mode 100644 arch/arm/mach-omap2/board-h4-mmc.c diff --git a/arch/arm/mach-omap2/Makefile b/arch/arm/mach-omap2/Makefile index 45b85fc..842f9fb 100644 --- a/arch/arm/mach-omap2/Makefile +++ b/arch/arm/mach-omap2/Makefile @@ -21,7 +21,7 @@ mmu_mach-objs := mmu.o # Specific board support obj-$(CONFIG_MACH_OMAP_GENERIC) += board-generic.o -obj-$(CONFIG_MACH_OMAP_H4) += board-h4.o +obj-$(CONFIG_MACH_OMAP_H4) += board-h4.o board-h4-mmc.o obj-$(CONFIG_MACH_OMAP_2430SDP) += board-2430sdp.o \ board-2430sdp-flash.o \ board-sdp-hsmmc.o \ diff --git a/arch/arm/mach-omap2/board-h4-mmc.c b/arch/arm/mach-omap2/board-h4-mmc.c new file mode 100644 index 0000000..9cdb00f --- /dev/null +++ b/arch/arm/mach-omap2/board-h4-mmc.c @@ -0,0 +1,266 @@ +/* + * linux/arch/arm/mach-omap2/board-h4-mmc.c + * + * Copyright (C) 2007 Instituto Nokia de Tecnologia - INdT + * Authors: David Cohen <david.cohen@xxxxxxxxxxx> + * Carlos Eduardo Aguiar <carlos.aguiar@xxxxxxxxxxx> + * + * This code is based on linux/arch/arm/mach-omap2/board-n800-mmc.c, which is: + * Copyright (C) 2006 Nokia Corporation + * Author: Juha Yrjola + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License version 2 as + * published by the Free Software Foundation. + */ + +#include <asm/arch/mmc.h> +#include <asm/arch/menelaus.h> + +#include <asm/mach-types.h> +#include <linux/delay.h> + +#ifdef CONFIG_MMC_OMAP + +/* Bit mask for slots detection interrupts */ +#define SD1_CD_ST (1 << 0) +#define SD2_CD_ST (1 << 1) + +static int slot1_cover_open; +static int slot2_cover_open; +static struct device *mmc_device; + +/* + * VMMC --> slot 1 + * VDCDC3_APE, VMCS2_APE --> slot 2 + */ + +static int h4_mmc_switch_slot(struct device *dev, int slot) +{ + int r = 0; + +#ifdef CONFIG_MMC_DEBUG + dev_dbg(dev, "Choose slot %d\n", slot + 1); +#endif + if (slot == 0) { + r = menelaus_enable_slot(2, 0); + r |= menelaus_enable_slot(1, 1); + } else { + r = menelaus_enable_slot(1, 0); + r |= menelaus_enable_slot(2, 1); + } + + return r ? -ENODEV : 0; +} + +static int h4_mmc_set_power(struct device *dev, int slot, int power_on, + int vdd) +{ + int mV = 0; + +#ifdef CONFIG_MMC_DEBUG + dev_dbg(dev, "Set slot %d power: %s (vdd %d)\n", slot + 1, + power_on ? "on" : "off", vdd); +#endif + if (slot == 0) { + if (!power_on) + return menelaus_set_vmmc(3000); + switch (1 << vdd) { + case MMC_VDD_33_34: + case MMC_VDD_32_33: + case MMC_VDD_31_32: + mV = 3100; + break; + case MMC_VDD_30_31: + mV = 3000; + break; + case MMC_VDD_28_29: + mV = 2800; + break; + case MMC_VDD_165_195: + mV = 1850; + break; + default: + BUG(); + } + return menelaus_set_vmmc(mV); + } else { + if (!power_on) + return menelaus_set_vdcdc(3, 3000); + switch (1 << vdd) { + case MMC_VDD_33_34: + case MMC_VDD_32_33: + mV = 3300; + break; + case MMC_VDD_30_31: + case MMC_VDD_29_30: + mV = 3000; + break; + case MMC_VDD_28_29: + case MMC_VDD_27_28: + mV = 2800; + break; + case MMC_VDD_24_25: + case MMC_VDD_23_24: + mV = 2400; + break; + case MMC_VDD_22_23: + case MMC_VDD_21_22: + mV = 2200; + break; + case MMC_VDD_20_21: + mV = 2000; + break; + case MMC_VDD_165_195: + mV = 1800; + break; + default: + BUG(); + } + return menelaus_set_vdcdc(3, mV); + } + return 0; +} + +static int h4_mmc_set_bus_mode(struct device *dev, int slot, int bus_mode) +{ + int r = 0; + +#ifdef CONFIG_MMC_DEBUG + dev_dbg(dev, "Set slot %d bus mode %s\n", slot + 1, + bus_mode == MMC_BUSMODE_OPENDRAIN ? "open-drain" : "push-pull"); +#endif + BUG_ON(slot != 0 && slot != 1); + slot++; + switch (bus_mode) { + case MMC_BUSMODE_OPENDRAIN: + r = menelaus_set_mmc_opendrain(slot, 1); + break; + case MMC_BUSMODE_PUSHPULL: + r = menelaus_set_mmc_opendrain(slot, 0); + break; + default: + BUG(); + } + if (r != 0 && printk_ratelimit()) { + dev_err(dev, "MMC: unable to set bus mode for slot %d\n", + slot); + } + return r; +} + +static int h4_mmc_slot1_cover_state(struct device *dev, int slot) +{ + BUG_ON(slot != 0); + return slot1_cover_open; +} + +static int h4_mmc_slot2_cover_state(struct device *dev, int slot) +{ + BUG_ON(slot != 1); + return slot2_cover_open; +} + +static void h4_mmc_slot_callback(void *data, u8 card_mask) +{ + int cover_open; + + cover_open = (card_mask & SD1_CD_ST) ? 0 : 1; + if (cover_open != slot1_cover_open) { + slot1_cover_open = cover_open; + omap_mmc_notify_cover_event(mmc_device, 0, slot1_cover_open); + } + + cover_open = (card_mask & SD2_CD_ST) ? 0 : 1; + if (cover_open != slot2_cover_open) { + slot2_cover_open = cover_open; + omap_mmc_notify_cover_event(mmc_device, 1, slot2_cover_open); + } +} + +static int h4_mmc_late_init(struct device *dev) +{ + int r; + + mmc_device = dev; + + r = menelaus_set_mmc_slot(1, 0, 0, 1); + if (r < 0) + goto out; + r = menelaus_set_mmc_slot(2, 0, 0, 1); + if (r < 0) + goto out; + + r = menelaus_get_slot_pin_states(); + if (r < 0) + goto out; + + if (r & SD1_CD_ST) + slot1_cover_open = 1; + else + slot1_cover_open = 0; + + /* Slot pin bits seem to be inversed until first swith change, + * but just for slot 2 + */ + if ((r == 0xf) || (r == (0xf & ~SD2_CD_ST))) + r = ~r; + + if (r & SD2_CD_ST) + slot2_cover_open = 1; + else + slot2_cover_open = 0; + + r = menelaus_register_mmc_callback(h4_mmc_slot_callback, NULL); + +out: + return r; +} + +static void h4_mmc_cleanup(struct device *dev) +{ + menelaus_unregister_mmc_callback(); +} + +static struct omap_mmc_platform_data h4_mmc_data = { + .nr_slots = 2, + .switch_slot = h4_mmc_switch_slot, + .init = h4_mmc_late_init, + .cleanup = h4_mmc_cleanup, + .slots[0] = { + .set_power = h4_mmc_set_power, + .set_bus_mode = h4_mmc_set_bus_mode, + .get_ro = NULL, + .get_cover_state= h4_mmc_slot1_cover_state, + .ocr_mask = MMC_VDD_165_195 | + MMC_VDD_28_29 | MMC_VDD_30_31 | + MMC_VDD_32_33 | MMC_VDD_33_34, + .name = "slot1", + }, + .slots[1] = { + .set_power = h4_mmc_set_power, + .set_bus_mode = h4_mmc_set_bus_mode, + .get_ro = NULL, + .get_cover_state= h4_mmc_slot2_cover_state, + .ocr_mask = MMC_VDD_165_195 | MMC_VDD_20_21 | + MMC_VDD_21_22 | MMC_VDD_22_23 | MMC_VDD_23_24 | + MMC_VDD_24_25 | MMC_VDD_27_28 | MMC_VDD_28_29 | + MMC_VDD_29_30 | MMC_VDD_30_31 | MMC_VDD_32_33 | + MMC_VDD_33_34, + .name = "slot2", + }, +}; + +void __init h4_mmc_init(void) +{ + omap_set_mmc_info(1, &h4_mmc_data); +} + +#else + +void __init h4_mmc_init(void) +{ +} + +#endif + diff --git a/arch/arm/mach-omap2/board-h4.c b/arch/arm/mach-omap2/board-h4.c index ef91d21..9b494d0 100644 --- a/arch/arm/mach-omap2/board-h4.c +++ b/arch/arm/mach-omap2/board-h4.c @@ -385,12 +385,15 @@ static struct omap_mmc_config h4_mmc_config __initdata = { .mmc [0] = { .enabled = 1, .wire4 = 1, - .wp_pin = -1, - .power_pin = -1, - .switch_pin = -1, + }, + .mmc [1] = { + .enabled = 1, + .wire4 = 1, }, }; +extern struct omap_mmc_platform_data h4_mmc_data; + static struct omap_lcd_config h4_lcd_config __initdata = { .ctrl_name = "internal", }; @@ -688,6 +691,7 @@ static void __init omap_h4_init(void) omap_board_config = h4_config; omap_board_config_size = ARRAY_SIZE(h4_config); omap_serial_init(); + h4_mmc_init(); omap_register_i2c_bus(1, 100, h4_i2c_board_info, ARRAY_SIZE(h4_i2c_board_info)); diff --git a/include/asm-arm/arch-omap/board-h4.h b/include/asm-arm/arch-omap/board-h4.h index 23b5ac6..9cc2ff7 100644 --- a/include/asm-arm/arch-omap/board-h4.h +++ b/include/asm-arm/arch-omap/board-h4.h @@ -29,6 +29,9 @@ #ifndef __ASM_ARCH_OMAP_H4_H #define __ASM_ARCH_OMAP_H4_H +/* MMC Prototypes */ +extern void h4_mmc_init(void); + /* Placeholder for H4 specific defines */ #define OMAP24XX_ETHR_GPIO_IRQ 92 #endif /* __ASM_ARCH_OMAP_H4_H */ -- 1.5.3.5 - To unsubscribe from this list: send the line "unsubscribe linux-omap" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html