summaryrefslogtreecommitdiff
path: root/package/broadcom-mmc/src/mmc.c
diff options
context:
space:
mode:
Diffstat (limited to 'package/broadcom-mmc/src/mmc.c')
-rw-r--r--package/broadcom-mmc/src/mmc.c676
1 files changed, 676 insertions, 0 deletions
diff --git a/package/broadcom-mmc/src/mmc.c b/package/broadcom-mmc/src/mmc.c
new file mode 100644
index 0000000..64cb8d9
--- /dev/null
+++ b/package/broadcom-mmc/src/mmc.c
@@ -0,0 +1,676 @@
+#include <linux/delay.h>
+#include <linux/timer.h>
+
+#include <linux/module.h>
+#include <linux/mm.h>
+#include <linux/init.h>
+#include <linux/fs.h>
+#include <linux/blkpg.h>
+#include <linux/hdreg.h>
+#include <linux/major.h>
+#include <asm/uaccess.h>
+#include <asm/io.h>
+
+#define DEVICE_NAME "mmc"
+#define DEVICE_NR(device) (MINOR(device))
+#define DEVICE_ON(device)
+#define DEVICE_OFF(device)
+#define MAJOR_NR 121
+
+#include <linux/blk.h>
+
+MODULE_AUTHOR("Madsuk/Rohde");
+MODULE_DESCRIPTION("Driver MMC/SD-Cards");
+MODULE_SUPPORTED_DEVICE("WRT54G");
+MODULE_LICENSE("GPL");
+
+#define SD_DI 0x20
+#define SD_DO 0x10
+#define SD_CLK 0x08
+#define SD_CS 0x80
+
+/* we have only one device */
+static int hd_sizes[1<<6];
+static int hd_blocksizes[1<<6];
+static int hd_hardsectsizes[1<<6];
+static int hd_maxsect[1<<6];
+static struct hd_struct hd[1<<6];
+
+static struct timer_list mmc_timer;
+static int mmc_media_detect = 0;
+static int mmc_media_changed = 1;
+
+typedef unsigned int uint32;
+
+static unsigned char port_state = 0x00;
+static volatile uint32 *gpioaddr_input = (uint32 *)0xb8000060;
+static volatile uint32 *gpioaddr_output = (uint32 *)0xb8000064;
+static volatile uint32 *gpioaddr_enable = (uint32 *)0xb8000068;
+static volatile uint32 *gpioaddr_control = (uint32 *)0xb800006c;
+
+static void mmc_spi_cs_low(void)
+{
+ port_state &= ~(SD_CS);
+ *gpioaddr_output = port_state;
+}
+
+static void mmc_spi_cs_high(void)
+{
+ port_state |= SD_CS;
+ *gpioaddr_output = port_state;
+}
+
+static unsigned char mmc_spi_io(unsigned char data_out)
+{
+ int i;
+ unsigned char result = 0, tmp_data = 0;
+
+ for(i=0; i<8; i++) {
+ if(data_out & (0x01 << (7-i)))
+ port_state |= SD_DI;
+ else
+ port_state &= ~SD_DI;
+
+ *gpioaddr_output = port_state;
+ port_state |= SD_CLK;
+ *gpioaddr_output = port_state;
+
+ tmp_data = *gpioaddr_input;
+
+ port_state &= ~SD_CLK;
+ *gpioaddr_output = port_state;
+
+ result <<= 1;
+
+ if(tmp_data & SD_DO)
+ result |= 1;
+ }
+
+ return(result);
+}
+
+static int mmc_write_block(unsigned int dest_addr, unsigned char *data)
+{
+ unsigned int address;
+ unsigned char r = 0;
+ unsigned char ab0, ab1, ab2, ab3;
+ int i;
+
+ address = dest_addr;
+
+ ab3 = 0xff & (address >> 24);
+ ab2 = 0xff & (address >> 16);
+ ab1 = 0xff & (address >> 8);
+ ab0 = 0xff & address;
+ mmc_spi_cs_low();
+ for (i = 0; i < 4; i++) mmc_spi_io(0xff);
+ mmc_spi_io(0x58);
+ mmc_spi_io(ab3); /* msb */
+ mmc_spi_io(ab2);
+ mmc_spi_io(ab1);
+ mmc_spi_io(ab0); /* lsb */
+ mmc_spi_io(0xff);
+ for (i = 0; i < 8; i++)
+ {
+ r = mmc_spi_io(0xff);
+ if (r == 0x00) break;
+ }
+ if (r != 0x00)
+ {
+ mmc_spi_cs_high();
+ mmc_spi_io(0xff);
+ return(1);
+ }
+
+ mmc_spi_io(0xfe);
+ for (i = 0; i < 512; i++) mmc_spi_io(data[i]);
+ for (i = 0; i < 2; i++) mmc_spi_io(0xff);
+
+ for (i = 0; i < 1000000; i++)
+ {
+ r = mmc_spi_io(0xff);
+ if (r == 0xff) break;
+ }
+ if (r != 0xff)
+ {
+ mmc_spi_cs_high();
+ mmc_spi_io(0xff);
+ return(3);
+ }
+ mmc_spi_cs_high();
+ mmc_spi_io(0xff);
+ return(0);
+}
+
+static int mmc_read_block(unsigned char *data, unsigned int src_addr)
+{
+ unsigned int address;
+ unsigned char r = 0;
+ unsigned char ab0, ab1, ab2, ab3;
+ int i;
+
+ address = src_addr;
+
+ ab3 = 0xff & (address >> 24);
+ ab2 = 0xff & (address >> 16);
+ ab1 = 0xff & (address >> 8);
+ ab0 = 0xff & address;
+
+ mmc_spi_cs_low();
+ for (i = 0; i < 4; i++) mmc_spi_io(0xff);
+ mmc_spi_io(0x51);
+ mmc_spi_io(ab3); /* msb */
+ mmc_spi_io(ab2);
+ mmc_spi_io(ab1);
+ mmc_spi_io(ab0); /* lsb */
+
+ mmc_spi_io(0xff);
+ for (i = 0; i < 8; i++)
+ {
+ r = mmc_spi_io(0xff);
+ if (r == 0x00) break;
+ }
+ if (r != 0x00)
+ {
+ mmc_spi_cs_high();
+ mmc_spi_io(0xff);
+ return(1);
+ }
+ for (i = 0; i < 100000; i++)
+ {
+ r = mmc_spi_io(0xff);
+ if (r == 0xfe) break;
+ }
+ if (r != 0xfe)
+ {
+ mmc_spi_cs_high();
+ mmc_spi_io(0xff);
+ return(2);
+ }
+ for (i = 0; i < 512; i++)
+ {
+ r = mmc_spi_io(0xff);
+ data[i] = r;
+ }
+ for (i = 0; i < 2; i++)
+ {
+ r = mmc_spi_io(0xff);
+ }
+ mmc_spi_cs_high();
+ mmc_spi_io(0xff);
+
+ return(0);
+}
+
+static void mmc_request(request_queue_t *q)
+{
+ unsigned int mmc_address;
+ unsigned char *buffer_address;
+ int nr_sectors;
+ int i;
+ int cmd;
+ int rc, code;
+
+ (void)q;
+ while (1)
+ {
+ code = 1; // Default is success
+ INIT_REQUEST;
+ mmc_address = (CURRENT->sector + hd[MINOR(CURRENT->rq_dev)].start_sect) * hd_hardsectsizes[0];
+ buffer_address = CURRENT->buffer;
+ nr_sectors = CURRENT->current_nr_sectors;
+ cmd = CURRENT->cmd;
+ if (((CURRENT->sector + CURRENT->current_nr_sectors + hd[MINOR(CURRENT->rq_dev)].start_sect) > hd[0].nr_sects) || (mmc_media_detect == 0))
+ {
+ code = 0;
+ }
+ else if (cmd == READ)
+ {
+ spin_unlock_irq(&io_request_lock);
+ for (i = 0; i < nr_sectors; i++)
+ {
+ rc = mmc_read_block(buffer_address, mmc_address);
+ if (rc != 0)
+ {
+ printk("mmc: error in mmc_read_block (%d)\n", rc);
+ code = 0;
+ break;
+ }
+ else
+ {
+ mmc_address += hd_hardsectsizes[0];
+ buffer_address += hd_hardsectsizes[0];
+ }
+ }
+ spin_lock_irq(&io_request_lock);
+ }
+ else if (cmd == WRITE)
+ {
+ spin_unlock_irq(&io_request_lock);
+ for (i = 0; i < nr_sectors; i++)
+ {
+ rc = mmc_write_block(mmc_address, buffer_address);
+ if (rc != 0)
+ {
+ printk("mmc: error in mmc_write_block (%d)\n", rc);
+ code = 0;
+ break;
+ }
+ else
+ {
+ mmc_address += hd_hardsectsizes[0];
+ buffer_address += hd_hardsectsizes[0];
+ }
+ }
+ spin_lock_irq(&io_request_lock);
+ }
+ else
+ {
+ code = 0;
+ }
+ end_request(code);
+ }
+}
+
+
+static int mmc_open(struct inode *inode, struct file *filp)
+{
+ int device;
+ (void)filp;
+
+ if (mmc_media_detect == 0) return -ENODEV;
+
+#if defined(MODULE)
+ MOD_INC_USE_COUNT;
+#endif
+ return 0;
+}
+
+static int mmc_release(struct inode *inode, struct file *filp)
+{
+ (void)filp;
+ fsync_dev(inode->i_rdev);
+ invalidate_buffers(inode->i_rdev);
+
+#if defined(MODULE)
+ MOD_DEC_USE_COUNT;
+#endif
+ return 0;
+}
+
+extern struct gendisk hd_gendisk;
+static int mmc_revalidate(kdev_t dev)
+{
+ int target, max_p, start, i;
+ if (mmc_media_detect == 0) return -ENODEV;
+
+ target = DEVICE_NR(dev);
+
+ max_p = hd_gendisk.max_p;
+ start = target << 6;
+ for (i = max_p - 1; i >= 0; i--) {
+ int minor = start + i;
+ invalidate_device(MKDEV(MAJOR_NR, minor), 1);
+ hd_gendisk.part[minor].start_sect = 0;
+ hd_gendisk.part[minor].nr_sects = 0;
+ }
+
+ grok_partitions(&hd_gendisk, target, 1 << 6,
+ hd_sizes[0] * 2);
+
+ return 0;
+}
+
+static int mmc_ioctl(struct inode *inode, struct file *filp, unsigned int cmd, unsigned long arg)
+{
+ if (!inode || !inode->i_rdev)
+ return -EINVAL;
+
+ switch(cmd) {
+ case BLKGETSIZE:
+ return put_user(hd[MINOR(inode->i_rdev)].nr_sects, (unsigned long *)arg);
+ case BLKGETSIZE64:
+ return put_user((u64)hd[MINOR(inode->i_rdev)].
+ nr_sects, (u64 *) arg);
+ case BLKRRPART:
+ if (!capable(CAP_SYS_ADMIN))
+ return -EACCES;
+
+ return mmc_revalidate(inode->i_rdev);
+ case HDIO_GETGEO:
+ {
+ struct hd_geometry *loc, g;
+ loc = (struct hd_geometry *) arg;
+ if (!loc)
+ return -EINVAL;
+ g.heads = 4;
+ g.sectors = 16;
+ g.cylinders = hd[0].nr_sects / (4 * 16);
+ g.start = hd[MINOR(inode->i_rdev)].start_sect;
+ return copy_to_user(loc, &g, sizeof(g)) ? -EFAULT : 0;
+ }
+ default:
+ return blk_ioctl(inode->i_rdev, cmd, arg);
+ }
+}
+
+static int mmc_card_init(void)
+{
+ unsigned char r = 0;
+ short i, j;
+ unsigned long flags;
+
+ save_flags(flags);
+ cli();
+
+ printk("mmc Card init\n");
+ mmc_spi_cs_high();
+ for (i = 0; i < 20; i++) mmc_spi_io(0xff);
+
+ mmc_spi_cs_low();
+
+ mmc_spi_io(0x40);
+ for (i = 0; i < 4; i++) mmc_spi_io(0x00);
+ mmc_spi_io(0x95);
+ for (i = 0; i < 8; i++)
+ {
+ r = mmc_spi_io(0xff);
+ if (r == 0x01) break;
+ }
+ mmc_spi_cs_high();
+ mmc_spi_io(0xff);
+ if (r != 0x01)
+ {
+ restore_flags(flags);
+ return(1);
+ }
+
+ printk("mmc Card init *1*\n");
+ for (j = 0; j < 10000; j++)
+ {
+ mmc_spi_cs_low();
+
+ mmc_spi_io(0x41);
+ for (i = 0; i < 4; i++) mmc_spi_io(0x00);
+ mmc_spi_io(0xff);
+ for (i = 0; i < 8; i++)
+ {
+ r = mmc_spi_io(0xff);
+ if (r == 0x00) break;
+ }
+ mmc_spi_cs_high();
+ mmc_spi_io(0xff);
+ if (r == 0x00)
+ {
+ restore_flags(flags);
+ printk("mmc Card init *2*\n");
+ return(0);
+ }
+ }
+ restore_flags(flags);
+
+ return(2);
+}
+
+static int mmc_card_config(void)
+{
+ unsigned char r = 0;
+ short i;
+ unsigned char csd[32];
+ unsigned int c_size;
+ unsigned int c_size_mult;
+ unsigned int mult;
+ unsigned int read_bl_len;
+ unsigned int blocknr = 0;
+ unsigned int block_len = 0;
+ unsigned int size = 0;
+
+ mmc_spi_cs_low();
+ for (i = 0; i < 4; i++) mmc_spi_io(0xff);
+ mmc_spi_io(0x49);
+ for (i = 0; i < 4; i++) mmc_spi_io(0x00);
+ mmc_spi_io(0xff);
+ for (i = 0; i < 8; i++)
+ {
+ r = mmc_spi_io(0xff);
+ if (r == 0x00) break;
+ }
+ if (r != 0x00)
+ {
+ mmc_spi_cs_high();
+ mmc_spi_io(0xff);
+ return(1);
+ }
+ for (i = 0; i < 8; i++)
+ {
+ r = mmc_spi_io(0xff);
+ if (r == 0xfe) break;
+ }
+ if (r != 0xfe)
+ {
+ mmc_spi_cs_high();
+ mmc_spi_io(0xff);
+ return(2);
+ }
+ for (i = 0; i < 16; i++)
+ {
+ r = mmc_spi_io(0xff);
+ csd[i] = r;
+ }
+ for (i = 0; i < 2; i++)
+ {
+ r = mmc_spi_io(0xff);
+ }
+ mmc_spi_cs_high();
+ mmc_spi_io(0xff);
+ if (r == 0x00) return(3);
+
+ c_size = csd[8] + csd[7] * 256 + (csd[6] & 0x03) * 256 * 256;
+ c_size >>= 6;
+ c_size_mult = csd[10] + (csd[9] & 0x03) * 256;
+ c_size_mult >>= 7;
+ read_bl_len = csd[5] & 0x0f;
+ mult = 1;
+ mult <<= c_size_mult + 2;
+ blocknr = (c_size + 1) * mult;
+ block_len = 1;
+ block_len <<= read_bl_len;
+ size = block_len * blocknr;
+ size >>= 10;
+
+ for(i=0; i<(1<<6); i++) {
+ hd_blocksizes[i] = 1024;
+ hd_hardsectsizes[i] = block_len;
+ hd_maxsect[i] = 256;
+ }
+ hd_sizes[0] = size;
+ hd[0].nr_sects = blocknr;
+
+
+ printk("Size = %d, hardsectsize = %d, sectors = %d\n",
+ size, block_len, blocknr);
+
+ return 0;
+}
+
+static int mmc_hardware_init(void)
+{
+ unsigned char gpio_outen;
+
+ // Set inputs/outputs here
+ printk("mmc Hardware init\n");
+ gpio_outen = *gpioaddr_enable;
+
+ gpio_outen = (gpio_outen | SD_DI | SD_CLK | SD_CS) & ~SD_DO;
+ *gpioaddr_enable = gpio_outen;
+
+ port_state = *gpioaddr_input;
+
+ // Clock low
+ port_state &= ~(SD_CLK | SD_DI | SD_CS);
+ *gpioaddr_output = port_state;
+
+ return 0;
+}
+
+static int mmc_check_media_change(kdev_t dev)
+{
+ (void)dev;
+ if (mmc_media_changed == 1)
+ {
+ mmc_media_changed = 0;
+ return 1;
+ }
+ else return 0;
+}
+
+static struct block_device_operations mmc_bdops =
+{
+ open: mmc_open,
+ release: mmc_release,
+ ioctl: mmc_ioctl,
+#if 0
+ check_media_change: mmc_check_media_change,
+ revalidate: mmc_revalidate,
+#endif
+};
+
+static struct gendisk hd_gendisk = {
+ major: MAJOR_NR,
+ major_name: DEVICE_NAME,
+ minor_shift: 6,
+ max_p: 1 << 6,
+ part: hd,
+ sizes: hd_sizes,
+ fops: &mmc_bdops,
+};
+
+static int mmc_init(void)
+{
+ int rc;
+
+ rc = mmc_hardware_init();
+
+ if ( rc != 0)
+ {
+ printk("mmc: error in mmc_hardware_init (%d)\n", rc);
+ return -1;
+ }
+
+ rc = mmc_card_init();
+ if ( rc != 0)
+ {
+ // Give it an extra shot
+ rc = mmc_card_init();
+ if ( rc != 0)
+ {
+ printk("mmc: error in mmc_card_init (%d)\n", rc);
+ return -1;
+ }
+ }
+
+ memset(hd_sizes, 0, sizeof(hd_sizes));
+ rc = mmc_card_config();
+ if ( rc != 0)
+ {
+ printk("mmc: error in mmc_card_config (%d)\n", rc);
+ return -1;
+ }
+
+
+ blk_size[MAJOR_NR] = hd_sizes;
+
+ memset(hd, 0, sizeof(hd));
+ hd[0].nr_sects = hd_sizes[0]*2;
+
+ blksize_size[MAJOR_NR] = hd_blocksizes;
+ hardsect_size[MAJOR_NR] = hd_hardsectsizes;
+ max_sectors[MAJOR_NR] = hd_maxsect;
+
+ hd_gendisk.nr_real = 1;
+
+ register_disk(&hd_gendisk, MKDEV(MAJOR_NR,0), 1<<6,
+ &mmc_bdops, hd_sizes[0]*2);
+
+ return 0;
+}
+
+static void mmc_exit(void)
+{
+ blk_size[MAJOR_NR] = NULL;
+ blksize_size[MAJOR_NR] = NULL;
+ hardsect_size[MAJOR_NR] = NULL;
+ max_sectors[MAJOR_NR] = NULL;
+ hd[0].nr_sects = 0;
+}
+
+static void mmc_check_media(void)
+{
+ int old_state;
+ int rc;
+
+ old_state = mmc_media_detect;
+
+ // TODO: Add card detection here
+ mmc_media_detect = 1;
+ if (old_state != mmc_media_detect)
+ {
+ mmc_media_changed = 1;
+ if (mmc_media_detect == 1)
+ {
+ rc = mmc_init();
+ if (rc != 0) printk("mmc: error in mmc_init (%d)\n", rc);
+ }
+ else
+ {
+ mmc_exit();
+ }
+ }
+
+ /* del_timer(&mmc_timer);
+ mmc_timer.expires = jiffies + 10*HZ;
+ add_timer(&mmc_timer); */
+}
+
+static int __init mmc_driver_init(void)
+{
+ int rc;
+
+ rc = devfs_register_blkdev(MAJOR_NR, DEVICE_NAME, &mmc_bdops);
+ if (rc < 0)
+ {
+ printk(KERN_WARNING "mmc: can't get major %d\n", MAJOR_NR);
+ return rc;
+ }
+
+ blk_init_queue(BLK_DEFAULT_QUEUE(MAJOR_NR), mmc_request);
+
+ read_ahead[MAJOR_NR] = 8;
+ add_gendisk(&hd_gendisk);
+
+ mmc_check_media();
+
+ /*init_timer(&mmc_timer);
+ mmc_timer.expires = jiffies + HZ;
+ mmc_timer.function = (void *)mmc_check_media;
+ add_timer(&mmc_timer);*/
+
+ return 0;
+}
+
+static void __exit mmc_driver_exit(void)
+{
+ int i;
+ del_timer(&mmc_timer);
+
+ for (i = 0; i < (1 << 6); i++)
+ fsync_dev(MKDEV(MAJOR_NR, i));
+
+ blk_cleanup_queue(BLK_DEFAULT_QUEUE(MAJOR_NR));
+ del_gendisk(&hd_gendisk);
+ devfs_unregister_blkdev(MAJOR_NR, DEVICE_NAME);
+ mmc_exit();
+}
+
+module_init(mmc_driver_init);
+module_exit(mmc_driver_exit);