Add Broadcom / Netgear changes from RAXE 1.0.0.48
[project/bcm63xx/u-boot.git] / drivers / mmc / bcmbca_sdhci_spl.c
diff --git a/drivers/mmc/bcmbca_sdhci_spl.c b/drivers/mmc/bcmbca_sdhci_spl.c
new file mode 100644 (file)
index 0000000..191b4af
--- /dev/null
@@ -0,0 +1,80 @@
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * (C) Copyright 2019  Broadcom Ltd
+ *
+ * Author: Farhan Ali <farhan.ali@broadcom.com>
+ */
+
+#include <common.h>
+#include <mmc.h>
+#include <malloc.h>
+
+char temp_buff[512];
+
+/* Copy over data from a specified offset */
+void mmc_spl_load_image(uint32_t offs, unsigned int size, void *vdst)
+{
+       uint blk_start, blk_cnt, err;
+       char * datap = vdst;
+       int i=0;
+       uint init_offset = 0;
+       uint orig_size = size;
+
+       struct mmc *mmc = find_mmc_device(0);
+       struct blk_desc * block_dev = mmc_get_blk_desc(mmc);
+       if (!mmc) {
+               puts("spl: mmc device not found!!\n");
+               hang();
+       }
+
+       if (mmc_init(mmc)) {
+               puts("MMC init failed\n");
+               return;
+       }
+       
+       if( mmc->read_bl_len > sizeof(temp_buff) )
+       {
+               puts("MMC temp buffer is smaller than read_bl_len\n");
+               hang();
+       }
+
+       blk_start = offs/mmc->read_bl_len;
+       init_offset = offs % mmc->read_bl_len;
+       blk_cnt = (init_offset+size)/mmc->read_bl_len + ((init_offset+size)%mmc->read_bl_len?1:0);
+
+       debug("Addr:0x%08x Len:0x%08x Startb:0x%08x Numb:0x%08x Endb:0x%08x\n"
+               , offs, size, blk_start, blk_cnt                        
+               ,((offs+size)/mmc->read_bl_len + ((offs+size)%mmc->read_bl_len?1:0)));
+
+       for( i=0; i<blk_cnt; i++ )
+       {
+               err = blk_dread(block_dev, blk_start, 1,
+                               temp_buff);
+               if (err != 1) 
+               {
+                       puts("spl: mmc read failed!!\n");
+                       hang();
+               } 
+               else
+               {
+                       /* All data in current block, copy it over */
+                       if( (init_offset + size) < mmc->read_bl_len )
+                       {
+                               memcpy(datap, temp_buff+init_offset, size);
+                               size = 0;
+                               init_offset = 0;
+                       }
+                       else
+                       {
+                               /* Data spans blocks, copy all data in this block */
+                               memcpy(datap, temp_buff+init_offset, mmc->read_bl_len-init_offset);
+                               size -= mmc->read_bl_len-init_offset;
+                               datap += mmc->read_bl_len-init_offset;
+                               init_offset = 0;
+                       }
+
+                       blk_start++;
+               }
+       }
+}
+