upstream u-boot with additional patches for our devices/boards:
https://lists.denx.de/pipermail/u-boot/2017-March/282789.html (AXP crashes) ;
Gbit ethernet patch for some LIME2 revisions ;
with SPI flash support
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
158 lines
3.8 KiB
158 lines
3.8 KiB
18 years ago
|
/*
|
||
|
* This program is free software; you can redistribute it and/or
|
||
|
* modify it under the terms of the GNU General Public License as
|
||
|
* published by the Free Software Foundation; either version 2 of
|
||
|
* the License, or (at your option) any later version.
|
||
|
*
|
||
|
* This program is distributed in the hope that it will be useful,
|
||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
* GNU General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License
|
||
|
* along with this program; if not, write to the Free Software
|
||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
|
||
|
* MA 02111-1307 USA
|
||
|
*/
|
||
|
|
||
|
#include <config.h>
|
||
|
#include <common.h>
|
||
|
#include <mmc.h>
|
||
|
#include <asm/errno.h>
|
||
|
#include <asm/arch/hardware.h>
|
||
|
#include <part.h>
|
||
|
#include <fat.h>
|
||
|
#include "mmc_hw.h"
|
||
|
#include <asm/arch/spi.h>
|
||
|
|
||
|
#ifdef CONFIG_MMC
|
||
|
|
||
|
#undef MMC_DEBUG
|
||
|
|
||
|
static block_dev_desc_t mmc_dev;
|
||
|
|
||
|
/* these are filled out by a call to mmc_hw_get_parameters */
|
||
|
static int hw_size; /* in kbytes */
|
||
|
static int hw_nr_sects;
|
||
|
static int hw_sect_size; /* in bytes */
|
||
|
|
||
|
block_dev_desc_t * mmc_get_dev(int dev)
|
||
|
{
|
||
|
return (block_dev_desc_t *)(&mmc_dev);
|
||
|
}
|
||
|
|
||
|
unsigned long mmc_block_read(int dev,
|
||
|
unsigned long start,
|
||
|
lbaint_t blkcnt,
|
||
|
void *buffer)
|
||
|
{
|
||
|
unsigned long rc = 0;
|
||
|
unsigned char *p = (unsigned char *)buffer;
|
||
|
unsigned long i;
|
||
|
unsigned long addr = start;
|
||
|
|
||
|
#ifdef MMC_DEBUG
|
||
|
printf("mmc_block_read: start=%lu, blkcnt=%lu\n", start,
|
||
|
(unsigned long)blkcnt);
|
||
|
#endif
|
||
|
|
||
|
for(i = 0; i < (unsigned long)blkcnt; i++) {
|
||
|
#ifdef MMC_DEBUG
|
||
|
printf("mmc_read_sector: addr=%lu, buffer=%p\n", addr, p);
|
||
|
#endif
|
||
|
(void)mmc_read_sector(addr, p);
|
||
|
rc++;
|
||
|
addr++;
|
||
|
p += hw_sect_size;
|
||
|
}
|
||
|
|
||
|
return rc;
|
||
|
}
|
||
|
|
||
|
/*-----------------------------------------------------------------------------
|
||
|
* Read hardware paramterers (sector size, size, number of sectors)
|
||
|
*/
|
||
|
static int mmc_hw_get_parameters(void)
|
||
|
{
|
||
|
unsigned char csddata[16];
|
||
|
unsigned int sizemult;
|
||
|
unsigned int size;
|
||
|
|
||
|
mmc_read_csd(csddata);
|
||
|
hw_sect_size = 1<<(csddata[5] & 0x0f);
|
||
|
size = ((csddata[6]&0x03)<<10)+(csddata[7]<<2)+(csddata[8]&0xc0);
|
||
|
sizemult = ((csddata[10] & 0x80)>>7)+((csddata[9] & 0x03)<<1);
|
||
|
hw_nr_sects = (size+1)*(1<<(sizemult+2));
|
||
|
hw_size = hw_nr_sects*hw_sect_size/1024;
|
||
|
|
||
|
#ifdef MMC_DEBUG
|
||
|
printf("mmc_hw_get_parameters: hw_sect_size=%d, hw_nr_sects=%d, "
|
||
|
"hw_size=%d\n", hw_sect_size, hw_nr_sects, hw_size);
|
||
|
#endif
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int mmc_init(int verbose)
|
||
|
{
|
||
|
int ret = -ENODEV;
|
||
|
|
||
|
if (verbose)
|
||
|
printf("mmc_init\n");
|
||
|
|
||
|
spi_init();
|
||
|
/* this meeds to be done twice */
|
||
|
mmc_hw_init();
|
||
|
udelay(1000);
|
||
|
mmc_hw_init();
|
||
|
|
||
|
mmc_hw_get_parameters();
|
||
|
|
||
|
mmc_dev.if_type = IF_TYPE_MMC;
|
||
|
mmc_dev.part_type = PART_TYPE_DOS;
|
||
|
mmc_dev.dev = 0;
|
||
|
mmc_dev.lun = 0;
|
||
|
mmc_dev.type = 0;
|
||
|
mmc_dev.blksz = hw_sect_size;
|
||
|
mmc_dev.lba = hw_nr_sects;
|
||
|
sprintf((char*)mmc_dev.vendor, "Unknown vendor");
|
||
|
sprintf((char*)mmc_dev.product, "Unknown product");
|
||
|
sprintf((char*)mmc_dev.revision, "N/A");
|
||
|
mmc_dev.removable = 0; /* should be true??? */
|
||
|
mmc_dev.block_read = mmc_block_read;
|
||
|
|
||
|
fat_register_device(&mmc_dev, 1);
|
||
|
|
||
|
ret = 0;
|
||
|
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
int mmc_write(uchar * src, ulong dst, int size)
|
||
|
{
|
||
|
#ifdef MMC_DEBUG
|
||
|
printf("mmc_write: src=%p, dst=%lu, size=%u\n", src, dst, size);
|
||
|
#endif
|
||
|
/* Since mmc2info always returns 0 this function will never be called */
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int mmc_read(ulong src, uchar * dst, int size)
|
||
|
{
|
||
|
#ifdef MMC_DEBUG
|
||
|
printf("mmc_read: src=%lu, dst=%p, size=%u\n", src, dst, size);
|
||
|
#endif
|
||
|
/* Since mmc2info always returns 0 this function will never be called */
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int mmc2info(ulong addr)
|
||
|
{
|
||
|
/* This function is used by cmd_cp to determine if source or destination
|
||
|
address resides on MMC-card or not. We do not support copy to and from
|
||
|
MMC-card so we always return 0. */
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
#endif /* CONFIG_MMC */
|