diff options
| author | Wolfgang Denk <wd@denx.de> | 2012-03-30 18:09:08 +0200 | 
|---|---|---|
| committer | Wolfgang Denk <wd@denx.de> | 2012-03-30 18:09:08 +0200 | 
| commit | bc6f6c87b685bcdcd5bef522982d15209b6b9601 (patch) | |
| tree | e5f924a962f002a1015e157a54450dfa9b953e9e /drivers/mmc/mmc.c | |
| parent | f2ea62474b4da9fc41735cbc1fe8491b247e0930 (diff) | |
| parent | 4a0764858b0bdcb3508f01b96e3fa32b16cdb30f (diff) | |
| download | olio-uboot-2014.01-bc6f6c87b685bcdcd5bef522982d15209b6b9601.tar.xz olio-uboot-2014.01-bc6f6c87b685bcdcd5bef522982d15209b6b9601.zip | |
Merge branch 'master' of git://git.denx.de/u-boot-arm
* 'master' of git://git.denx.de/u-boot-arm: (146 commits)
  arm: Use common .lds file where possible
  arm: add a common .lds link script
  arm: Remove unneeded setting of LDCSRIPT
  Define CPUDIR for the .lds link script
  arm: Remove zipitz2 link script
  Allow arch directory to contain .lds without requiring Makefile
  OMAP: Remove omap1610inn-based boards
  arch/arm/cpu/armv7/omap-common/clocks-common.c: Fix build warnings
  board/ti/beagle/beagle.c: Fix build warnings
  sdrc.c: Fix typo in do_sdrc_init() for SPL
  tegra: i2c: Add I2C driver
  tegra: fdt: i2c: Add extra I2C bindings for U-Boot
  tegra: i2c: Select I2C ordering for Seaboard
  tegra: i2c: Enable I2C on Seaboard
  tegra: i2c: Select number of controllers for Tegra2 boards
  tegra: i2c: Initialise I2C on Nvidia boards
  tegra: Enhance clock support to handle 16-bit clock divisors
  fdt: Add function to allow aliases to refer to multiple nodes
  tegra: Rename NV_PA_PMC_BASE to TEGRA2_PMC_BASE
  tegra: fdt: Enable FDT support for Ventana
  tegra: fdt: Enable FDT support for Seaboard
  tegra: usb: Enable USB on Seaboard
  tegra: usb: Add common USB defines for tegra2 boards
  tegra: usb: Add USB support to nvidia boards
  arm: Check for valid FDT after console is up
  fdt: Avoid early panic() when there is no FDT present
  tegra: usb: Add support for Tegra USB peripheral
  tegra: fdt: Add function to return peripheral/clock ID
  usb: Add support for txfifo threshold
  tegra: usb: fdt: Add USB definitions for Tegra2 Seaboard
  tegra: usb: fdt: Add additional device tree definitions for USB ports
  tegra: fdt: Add clock bindings for Tegra2 Seaboard
  tegra: fdt: Add clock bindings
  tegra: fdt: Add additional USB binding
  fdt: Add tegra-usb bindings file from linux
  fdt: Add staging area for device tree binding documentation
  tegra: fdt: Add device tree file for Tegra2 Seaboard from kernel
  tegra: fdt: Add Tegra2x device tree file from kernel
  arm: fdt: Add skeleton device tree file from kernel
  fdt: Add basic support for decoding GPIO definitions
  fdt: Add functions to access phandles, arrays and bools
  fdt: Tidy up a few fdtdec problems
  fdt: Add tests for fdtdec
  fdt: Add fdtdec_find_aliases() to deal with alias nodes
  arm: Tegra2: Fix ELDK42 gcc failure with inline asm stack pointer load
  net: fec_mxc: allow use with cache enabled
  net: force PKTALIGN to ARCH_DMA_MINALIGN
  i.MX28: Enable caches by default
  i.MX28: Make use of the bounce buffer
  i.MX28: Do data transfers via DMA in MMC driver
  MMC: Implement generic bounce buffer
  i.MX28: Add cache support to MXS NAND driver
  i.MX28: Add cache support into the APBH DMA driver
  ARM926EJS: Implement cache operations
  board/vpac270/onenand.c: Fix build errors
  nhk8815: fix build errors
  atmel-boards: add missing atmel_mci.h
  ARM: highbank: setup env from boot source register
  ARM: highbank: change env config to use nvram
  ARM: highbank: add reset support
  ARM: highbank: Add boot counter support
  ARM: highbank: change TEXT_BASE to 0x8000
  ARM: highbank: fix us_to_tick calculation
  ARM: highbank: add missing get_tbclk
  ARM: highbank: fix warning for calxedaxgmac_initialize
  net: calxedaxgmac: fix build due to missing __aligned definition
  EXYNOS: Add structure for Exynos4 DMC
  EXYNOS: SMDK5250: Support all 4 UARTs
  ARM: fix s3c2410 timer code
  ARM: davinci: fixes for cam_enc_4xx board
  omap3_spi: receive transmit mode
  calimain, enbw_cmc: Fix typo in comments
  Davinci: ea20: use gpio framework to access gpios
  OMAP3: mt_ventoux: sets its own mtdparts
  OMAP3: mt_ventoux: updated timing for FPGA
  twl4030: fix potential power supply handling issues
  NAND: TI: fix warnings in omap_gpmc.c
  cam_enc_4xx: Rename 'images' to 'imgs'
  arm: Add Prep subcommand support to bootm
  OMAP3: twister: add support to boot Linux from SPL
  SPL: call cleanup_before_linux() before booting Linux
  OMAP3: SPL: do not call I2C init if no I2C is set.
  Add cache functions to SPL for armv7
  devkit8000: Implement and activate direct OS boot
  omap/spl: change output of spl_parse_image_header
  omap-common/spl: Add linux boot to SPL
  devkit8000/spl: init GPMC for dm9000 in SPL
  omap-common: Add NAND SPL linux booting
  devkit8000: add config for spl command
  Add cmd_spl command
  mx53ard: Initialize return code with error
  mx53: Make PLL2 to be the parent of UART clock
  configs: imx: Use CONFIG_SF_DEFAULT_CS
  mx28evk: Provide default values for SPI bus and chip select
  USB: ehci-mx6: Add proper IO accessors
  mx6: Read silicon revision from register
  i.MX28: Drop __naked function from spl_mem_init
  mxs_spi: Return proper timeout error
  i.MX28: Make the stabilization delays shorter
  pmic_i2c: Return error in case of invalid pmic_i2c_tx_num
  mx6: Remove duplicate definition of ANATOP_BASE_ADDR
  mx6: Fix reset cause for Power On Reset case
  i.MX6: mx6qsabrelite: add MACH_TYPE_MX6Q_SABRELITE
  i.MX6: mx6q_sabrelite: add CONFIG_REVISION_TAG
  i.MX28: Enable additional DRAM address bits
  mx6q: mx6qsabrelite: setup_spi() should be called in board_init to allow use for environment
  mx31: add "ARM11P power gating" to get_reset_cause
  mx31pdk: Fix CONFIG_SYS_MEMTEST_END
  efikamx: Fix CONFIG_SYS_MEMTEST_END
  mx53smd: Fix CONFIG_SYS_MEMTEST_END
  mx53evk: Fix CONFIG_SYS_MEMTEST_END
  mx51evk: Fix CONFIG_SYS_MEMTEST_END
  i.MX6: mx6qsabrelite: add ext2 support
  imximage: Remove overwriting of flash_offset
  IXP: Fix GPIO_INT_ACT_LOW_SET()
  IXP: Fix NAND build warning on PDNB3 and SCPU
  IXP: Move PDNB3 and SCPU from Makefile to boards.cfg
  IXP: Squash warnings in IXP NPE
  IXP: Fix missing MACH_TYPE_{ACTUX?,PNB3,DVLHOST}
  IXP: Make IXP buildable with arm-linux- toolchains
  Examples: Properly append LDFLAGS to LD command
  SPL: Enable YMODEM support on BeagleBone and AM335x EVM
  SPL: Add YMODEM over UART load support
  SPL: Add README.omap3
  README: document more SPL config options
  spl.c: Use __noreturn decorator
  config.mk: Check for -fstack-usage support
  config.mk: Make cc-option create a file under include/generated
  ...
Diffstat (limited to 'drivers/mmc/mmc.c')
| -rw-r--r-- | drivers/mmc/mmc.c | 102 | 
1 files changed, 99 insertions, 3 deletions
| diff --git a/drivers/mmc/mmc.c b/drivers/mmc/mmc.c index 49c3349f5..74e5fea66 100644 --- a/drivers/mmc/mmc.c +++ b/drivers/mmc/mmc.c @@ -47,10 +47,105 @@ int __board_mmc_getcd(struct mmc *mmc) {  int board_mmc_getcd(struct mmc *mmc)__attribute__((weak,  	alias("__board_mmc_getcd"))); +#ifdef CONFIG_MMC_BOUNCE_BUFFER +static int mmc_bounce_need_bounce(struct mmc_data *orig) +{ +	ulong addr, len; + +	if (orig->flags & MMC_DATA_READ) +		addr = (ulong)orig->dest; +	else +		addr = (ulong)orig->src; + +	if (addr % ARCH_DMA_MINALIGN) { +		debug("MMC: Unaligned data destination address %08lx!\n", addr); +		return 1; +	} + +	len = (ulong)(orig->blocksize * orig->blocks); +	if (len % ARCH_DMA_MINALIGN) { +		debug("MMC: Unaligned data destination length %08lx!\n", len); +		return 1; +	} + +	return 0; +} + +static int mmc_bounce_buffer_start(struct mmc_data *backup, +					struct mmc_data *orig) +{ +	ulong origlen, len; +	void *buffer; + +	if (!orig) +		return 0; + +	if (!mmc_bounce_need_bounce(orig)) +		return 0; + +	memcpy(backup, orig, sizeof(struct mmc_data)); + +	origlen = orig->blocksize * orig->blocks; +	len = roundup(origlen, ARCH_DMA_MINALIGN); +	buffer = memalign(ARCH_DMA_MINALIGN, len); +	if (!buffer) { +		puts("MMC: Error allocating MMC bounce buffer!\n"); +		return 1; +	} + +	if (orig->flags & MMC_DATA_READ) { +		orig->dest = buffer; +	} else { +		memcpy(buffer, orig->src, origlen); +		orig->src = buffer; +	} + +	return 0; +} + +static void mmc_bounce_buffer_stop(struct mmc_data *backup, +					struct mmc_data *orig) +{ +	ulong len; + +	if (!orig) +		return; + +	if (!mmc_bounce_need_bounce(backup)) +		return; + +	if (backup->flags & MMC_DATA_READ) { +		len = backup->blocksize * backup->blocks; +		memcpy(backup->dest, orig->dest, len); +		free(orig->dest); +		orig->dest = backup->dest; +	} else { +		free((void *)orig->src); +		orig->src = backup->src; +	} + +	return; + +} +#else +static inline int mmc_bounce_buffer_start(struct mmc_data *backup, +					struct mmc_data *orig) { } +static inline void mmc_bounce_buffer_stop(struct mmc_data *backup, +					struct mmc_data *orig) { } +#endif +  int mmc_send_cmd(struct mmc *mmc, struct mmc_cmd *cmd, struct mmc_data *data)  { -#ifdef CONFIG_MMC_TRACE +	struct mmc_data backup;  	int ret; + +	memset(&backup, 0, sizeof(backup)); + +	ret = mmc_bounce_buffer_start(&backup, data); +	if (ret) +		return ret; + +#ifdef CONFIG_MMC_TRACE  	int i;  	u8 *ptr; @@ -99,10 +194,11 @@ int mmc_send_cmd(struct mmc *mmc, struct mmc_cmd *cmd, struct mmc_data *data)  			printf("\t\tERROR MMC rsp not supported\n");  			break;  	} -	return ret;  #else -	return mmc->send_cmd(mmc, cmd, data); +	ret = mmc->send_cmd(mmc, cmd, data);  #endif +	mmc_bounce_buffer_stop(&backup, data); +	return ret;  }  int mmc_send_status(struct mmc *mmc, int timeout) |