blob: a1c7a94a90e347c58c3e3b032e53870d6b83b15b [file] [log] [blame]
Tom Rini10e47792018-05-06 17:58:06 -04001// SPDX-License-Identifier: GPL-2.0+
Jerry Huang2c766ba2011-01-24 17:09:54 +00002/*
3 * Copyright 2011 Freescale Semiconductor, Inc.
Jerry Huang2c766ba2011-01-24 17:09:54 +00004 */
5
6#include <common.h>
7#include <mmc.h>
8#include <malloc.h>
9
10/*
11 * The environment variables are written to just after the u-boot image
12 * on SDCard, so we must read the MBR to get the start address and code
13 * length of the u-boot image, then calculate the address of the env.
14 */
15#define ESDHC_BOOT_IMAGE_SIZE 0x48
16#define ESDHC_BOOT_IMAGE_ADDR 0x50
17
Haijun.Zhang22e3c422014-01-10 13:52:19 +080018#define ESDHC_DEFAULT_ENVADDR 0x400
19
Michael Heimpolda1d25a32013-04-10 10:36:19 +000020int mmc_get_env_addr(struct mmc *mmc, int copy, u32 *env_addr)
Jerry Huang2c766ba2011-01-24 17:09:54 +000021{
22 u8 *tmp_buf;
23 u32 blklen, code_offset, code_len, n;
24
25 blklen = mmc->read_bl_len;
26 tmp_buf = malloc(blklen);
27 if (!tmp_buf)
28 return 1;
29
30 /* read out the first block, get the config data information */
Yinbo Zhuea3ff842019-10-15 17:20:49 +080031#ifdef CONFIG_BLK
32 n = blk_dread(mmc_get_blk_desc(mmc), 0, 1, tmp_buf);
33#else
Stephen Warrene73f2962015-12-07 11:38:48 -070034 n = mmc->block_dev.block_read(&mmc->block_dev, 0, 1, tmp_buf);
Yinbo Zhuea3ff842019-10-15 17:20:49 +080035#endif
Jerry Huang2c766ba2011-01-24 17:09:54 +000036 if (!n) {
37 free(tmp_buf);
38 return 1;
39 }
40
41 /* Get the Source Address, from offset 0x50 */
42 code_offset = *(u32 *)(tmp_buf + ESDHC_BOOT_IMAGE_ADDR);
43
44 /* Get the code size from offset 0x48 */
45 code_len = *(u32 *)(tmp_buf + ESDHC_BOOT_IMAGE_SIZE);
46
Haijun.Zhang22e3c422014-01-10 13:52:19 +080047#ifdef CONFIG_ESDHC_HC_BLK_ADDR
48 /*
49 * On soc BSC9131, BSC9132:
50 * In High Capacity SD Cards (> 2 GBytes), the 32-bit source address and
51 * code length of these soc specify the memory address in block address
52 * format. Block length is fixed to 512 bytes as per the SD High
53 * Capacity specification.
54 */
55 u64 tmp;
56
57 if (mmc->high_capacity) {
58 tmp = (u64)code_offset * blklen;
59 tmp += code_len * blklen;
60 } else
61 tmp = code_offset + code_len;
62
63 if ((tmp + CONFIG_ENV_SIZE > mmc->capacity) ||
64 (tmp > 0xFFFFFFFFU))
65 *env_addr = ESDHC_DEFAULT_ENVADDR;
66 else
67 *env_addr = tmp;
68
69 free(tmp_buf);
70
71 return 0;
72#endif
73
Jerry Huang2c766ba2011-01-24 17:09:54 +000074 *env_addr = code_offset + code_len;
75
76 free(tmp_buf);
77
78 return 0;
79}