diff --git a/bsp/cvitek/cv18xx_aarch64/.gitignore b/bsp/cvitek/cv18xx_aarch64/.gitignore new file mode 100755 index 0000000000..65b277bc26 --- /dev/null +++ b/bsp/cvitek/cv18xx_aarch64/.gitignore @@ -0,0 +1 @@ +Image \ No newline at end of file diff --git a/bsp/cvitek/cv18xx_aarch64/README.md b/bsp/cvitek/cv18xx_aarch64/README.md index 4d85cb2d79..a5ceb07dbf 100644 --- a/bsp/cvitek/cv18xx_aarch64/README.md +++ b/bsp/cvitek/cv18xx_aarch64/README.md @@ -63,7 +63,7 @@ scons -j12 ## 3. 运行 -> 目前仅支持使用uboot对 `rtthread.bin` 文件进行加载,uboot对 `boot.sd` 的加载方式后续会实现。 +### 3.1 uboot加载rtthread.bin 1. 将 SD 卡分为 2 个分区,第 1 个分区用于存放 bin 文件,第 2 个分区用于作为数据存储分区,分区格式为 `FAT32`。 @@ -81,6 +81,22 @@ go 0x80200000 > 0x80200000为rtthread.bin加载到内存的位置,可在menuconfig中自己修改,注意不能与小核固件加载位置重叠。 +### 3.2 uboot加载boot.sd + +1. 将 SD 卡分为 2 个分区,第 1 个分区用于存放 bin 文件,第 2 个分区用于作为数据存储分区,分区格式为 `FAT32`。 + +2. 将bsp的boot目录下的 `fip.bin` 和编译生成的 `boot.sd` 复制 SD 卡第一个分区中。后续更新固件只需要复制 `boot.sd` 文件即可。 + +配置**串口0**参数: 115200 8N1 ,硬件和软件流控为关。 + +直接上电运行,uboot会自动调用bootcmd解析boot.sd文件,然后加载`rtthread.bin`运行。 + +### 3.3 如何生成fip.bin + +在本bsp的boot/milkv-duo256m目录下存放了所有需要构建出fip.bin的一些依赖文件和相关脚本。用户只需要在boot目录下执行`combine.sh`即可生成fip.bin。 + +> 如何用户编译了小核c906_little的bsp,那么`combine.sh`脚本将会生成带有小核程序的fip.bin。未编译则不会。 + 完成后可以看到串口的输出信息: **标准版log信息:** diff --git a/bsp/cvitek/cv18xx_aarch64/boot/combine.sh b/bsp/cvitek/cv18xx_aarch64/boot/combine.sh new file mode 100755 index 0000000000..e39f49dcbc --- /dev/null +++ b/bsp/cvitek/cv18xx_aarch64/boot/combine.sh @@ -0,0 +1,44 @@ +C906_LITTLE_BIN_PATH=../../c906_little/rtthread.bin +GEN_FIP_PATH=. +DEPENDS_FILE_PATH=${GEN_FIP_PATH}/milkv-duo256m + +BLCP_IMG_RUNADDR=0x05200200 +BLCP_PARAM_LOADADDR=0 +NAND_INFO=00000000 +NOR_INFO='FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF' +FIP_COMPRESS=lzma + +CHIP_CONF_PATH=${DEPENDS_FILE_PATH}/chip_conf.bin +DDR_PARAM_TEST_PATH=${DEPENDS_FILE_PATH}/ddr_param.bin +BL2_PATH=${DEPENDS_FILE_PATH}/bl2.bin +BLCP_PATH=${DEPENDS_FILE_PATH}/empty.bin +MONITOR_PATH=${DEPENDS_FILE_PATH}/bl31.bin +LOADER_2ND_PATH=${DEPENDS_FILE_PATH}/u-boot-raw.bin + +if [ -f "$C906_LITTLE_BIN_PATH" ]; then + echo "The file $C906_LITTLE_BIN_PATH exist." + BLCP_2ND_PATH=${C906_LITTLE_BIN_PATH} +else + echo "The file $C906_LITTLE_BIN_PATH does not exist. Execute scons to compile it." +fi + +echo "Combining fip.bin..." +. ${DEPENDS_FILE_PATH}/blmacros.env && \ +${DEPENDS_FILE_PATH}/fiptool.py -v genfip \ +${GEN_FIP_PATH}/fip.bin \ +--MONITOR_RUNADDR="${MONITOR_RUNADDR}" \ +--BLCP_2ND_RUNADDR="${BLCP_2ND_RUNADDR}" \ +--CHIP_CONF=${CHIP_CONF_PATH} \ +--NOR_INFO=${NOR_INFO} \ +--NAND_INFO=${NAND_INFO} \ +--BL2=${BL2_PATH} \ +--BLCP_IMG_RUNADDR=${BLCP_IMG_RUNADDR} \ +--BLCP_PARAM_LOADADDR=${BLCP_PARAM_LOADADDR} \ +--BLCP=${BLCP_PATH} \ +--DDR_PARAM=${DDR_PARAM_TEST_PATH} \ +--BLCP_2ND=${BLCP_2ND_PATH} \ +--MONITOR=${MONITOR_PATH} \ +--LOADER_2ND=${LOADER_2ND_PATH} \ +--compress=${FIP_COMPRESS} + +echo "Combining fip.bin done!" diff --git a/bsp/cvitek/cv18xx_aarch64/boot/fip.bin b/bsp/cvitek/cv18xx_aarch64/boot/fip.bin index c406223010..be1eeeb042 100644 Binary files a/bsp/cvitek/cv18xx_aarch64/boot/fip.bin and b/bsp/cvitek/cv18xx_aarch64/boot/fip.bin differ diff --git a/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/bl2.bin b/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/bl2.bin new file mode 100755 index 0000000000..197e5a1f3a Binary files /dev/null and b/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/bl2.bin differ diff --git a/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/bl31.bin b/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/bl31.bin new file mode 100644 index 0000000000..cf38b2e291 Binary files /dev/null and b/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/bl31.bin differ diff --git a/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/blmacros.env b/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/blmacros.env new file mode 100644 index 0000000000..ae3059bd57 --- /dev/null +++ b/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/blmacros.env @@ -0,0 +1,2 @@ +MONITOR_RUNADDR=0x0000000080000000 +BLCP_2ND_RUNADDR=0x000000008fe00000 diff --git a/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/chip_conf.bin b/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/chip_conf.bin new file mode 100644 index 0000000000..d2dabd6868 Binary files /dev/null and b/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/chip_conf.bin differ diff --git a/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/ddr_param.bin b/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/ddr_param.bin new file mode 100644 index 0000000000..fb4c648fd7 Binary files /dev/null and b/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/ddr_param.bin differ diff --git a/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/empty.bin b/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/empty.bin new file mode 100644 index 0000000000..e69de29bb2 diff --git a/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/fiptool.py b/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/fiptool.py new file mode 100755 index 0000000000..6943e5f799 --- /dev/null +++ b/bsp/cvitek/cv18xx_aarch64/boot/milkv-duo256m/fiptool.py @@ -0,0 +1,803 @@ +#!/usr/bin/env python3 +# SPDX-License-Identifier: BSD-3-Clause +# PYTHON_ARGCOMPLETE_OK + +import sys +import logging +import os +import os.path +import argparse +from collections import OrderedDict +import binascii +from struct import pack, unpack +import lzma +import pprint + + +PYTHON_MIN_VERSION = (3, 5, 2) # Ubuntu 16.04 LTS contains Python v3.5.2 by default + + +if sys.version_info < PYTHON_MIN_VERSION: + print("Python >= %r is required" % (PYTHON_MIN_VERSION,)) + sys.exit(-1) + + +try: + import coloredlogs +except ImportError: + coloredlogs = None + +try: + import argcomplete +except ImportError: + argcomplete = None + + +LOADER_2ND_MAGIC_ORIG = b"BL33" +LOADER_2ND_MAGIC_LZMA = b"B3MA" +LOADER_2ND_MAGIC_LZ4 = b"B3Z4" + +LOADER_2ND_MAGIC_LIST = [ + LOADER_2ND_MAGIC_ORIG, + LOADER_2ND_MAGIC_LZMA, + LOADER_2ND_MAGIC_LZ4, +] + +IMAGE_ALIGN = 512 +PARAM1_SIZE = 0x1000 +PARAM1_SIZE_WO_SIG = 0x800 +PARAM2_SIZE = 0x1000 + + +def round_up(divident, divisor): + return ((divident + divisor - 1) // divisor) * divisor + + +def lzma_compress(body): + z = lzma.LZMACompressor(lzma.FORMAT_ALONE, preset=lzma.PRESET_EXTREME) + compressed = z.compress(body) + compressed += z.flush() + + return compressed + + +def lz4_compress(body): + try: + import lz4.frame + except ImportError: + logging.error("lz4 is not installed. Run 'pip install lz4'.") + raise + + compressed = lz4.frame.compress(body) + return compressed + + +class Entry: + __slots__ = "name", "type", "addr", "_content", "entry_size" + + def __init__(self): + self.addr = None + self._content = None + + @property + def end(self): + return self.addr + self.entry_size + + @property + def content(self): + return self._content + + @content.setter + def content(self, value): + if type(value) == int: + value = value.to_bytes(self.entry_size, "little") + + if self.entry_size is not None: + if len(value) > self.entry_size: + raise ValueError("%s (%d bytes) must <= %#r" % (self.name, len(value), self.entry_size)) + value = value + b"\0" * (self.entry_size - len(value)) + + self._content = value + + @classmethod + def make(cls, name, entry_size, _type, init=None): + entry = Entry() + entry.name = name + entry.type = _type + entry.entry_size = entry_size + + if type(init) in (bytes, bytearray): + entry.content = bytes(init) + elif entry_size is not None: + entry.content = b"\0" * entry.entry_size + else: + entry.content = b"" + + return (name, entry) + + def toint(self): + if self.type != int: + raise TypeError("%s is not int type" % self.name) + + return int.from_bytes(self.content, "little") + + def tostr(self): + v = self.content + if self.type == int: + v = "%#08x" % self.toint() + elif type(self.content) in [bytes, bytearray]: + v = v.hex() + if len(v) > 32: + v = v[:32] + "..." + + return v + + def __str__(self): + v = self.tostr() + return "<%s=%s (%dbytes)>" % (self.name, v, self.entry_size) + + def __repr__(self): + v = self.tostr() + return "<%s: a=%#x s=%#x c=%s %r>" % (self.name, self.addr, self.entry_size, v, self.type) + + +class FIP: + param1 = OrderedDict( + [ + Entry.make("MAGIC1", 8, int, b"CVBL01\n\0"), + Entry.make("MAGIC2", 4, int), + Entry.make("PARAM_CKSUM", 4, int), + Entry.make("NAND_INFO", 128, int), + Entry.make("NOR_INFO", 36, int), + Entry.make("FIP_FLAGS", 8, int), + Entry.make("CHIP_CONF_SIZE", 4, int), + Entry.make("BLCP_IMG_CKSUM", 4, int), + Entry.make("BLCP_IMG_SIZE", 4, int), + Entry.make("BLCP_IMG_RUNADDR", 4, int), + Entry.make("BLCP_PARAM_LOADADDR", 4, int), + Entry.make("BLCP_PARAM_SIZE", 4, int), + Entry.make("BL2_IMG_CKSUM", 4, int), + Entry.make("BL2_IMG_SIZE", 4, int), + Entry.make("BLD_IMG_SIZE", 4, int), + Entry.make("PARAM2_LOADADDR", 4, int), + Entry.make("RESERVED1", 4, int), + Entry.make("CHIP_CONF", 760, bytes), + Entry.make("BL_EK", 32, bytes), + Entry.make("ROOT_PK", 512, bytes), + Entry.make("BL_PK", 512, bytes), + Entry.make("BL_PK_SIG", 512, bytes), + Entry.make("CHIP_CONF_SIG", 512, bytes), + Entry.make("BL2_IMG_SIG", 512, bytes), + Entry.make("BLCP_IMG_SIG", 512, bytes), + ] + ) + + body1 = OrderedDict( + [ + Entry.make("BLCP", None, bytes), + Entry.make("BL2", None, bytes), + ] + ) + + param2 = OrderedDict( + [ + Entry.make("MAGIC1", 8, int, b"CVLD02\n\0"), + Entry.make("PARAM2_CKSUM", 4, int), + Entry.make("RESERVED1", 4, bytes), + # DDR param + Entry.make("DDR_PARAM_CKSUM", 4, int), + Entry.make("DDR_PARAM_LOADADDR", 4, int), + Entry.make("DDR_PARAM_SIZE", 4, int), + Entry.make("DDR_PARAM_RESERVED", 4, int), + # BLCP_2ND + Entry.make("BLCP_2ND_CKSUM", 4, int), + Entry.make("BLCP_2ND_LOADADDR", 4, int), + Entry.make("BLCP_2ND_SIZE", 4, int), + Entry.make("BLCP_2ND_RUNADDR", 4, int), + # ATF-BL31 or OpenSBI + Entry.make("MONITOR_CKSUM", 4, int), + Entry.make("MONITOR_LOADADDR", 4, int), + Entry.make("MONITOR_SIZE", 4, int), + Entry.make("MONITOR_RUNADDR", 4, int), + # u-boot + Entry.make("LOADER_2ND_RESERVED0", 4, int), + Entry.make("LOADER_2ND_LOADADDR", 4, int), + Entry.make("LOADER_2ND_RESERVED1", 4, int), + Entry.make("LOADER_2ND_RESERVED2", 4, int), + # Reserved + Entry.make("RESERVED_LAST", 4096 - 16 * 5, bytes), + ] + ) + + body2 = OrderedDict( + [ + Entry.make("DDR_PARAM", None, bytes), + Entry.make("BLCP_2ND", None, bytes), + Entry.make("MONITOR", None, bytes), + Entry.make("LOADER_2ND", None, bytes), + ] + ) + + ldr_2nd_hdr = OrderedDict( + [ + Entry.make("JUMP0", 4, int), + Entry.make("MAGIC", 4, int), + Entry.make("CKSUM", 4, int), + Entry.make("SIZE", 4, int), + Entry.make("RUNADDR", 8, int), + Entry.make("RESERVED1", 4, int), + Entry.make("RESERVED2", 4, int), + ] + ) + + FIP_FLAGS_SCS_MASK = 0x000c + FIP_FLAGS_ENCRYPTED_MASK = 0x0030 + + def _param_size(self, param): + return max((e.end for e in param.values())) + + def _gen_param(self): + addr = 0 + for entry in self.param1.values(): + entry.addr = addr + addr += entry.entry_size + + assert PARAM1_SIZE_WO_SIG == self.param1["BL_PK_SIG"].addr + + addr = 0 + for entry in self.param2.values(): + entry.addr = addr + addr += entry.entry_size + + assert PARAM2_SIZE == self.param2["RESERVED_LAST"].addr + self.param2["RESERVED_LAST"].entry_size + + addr = 0 + for entry in self.ldr_2nd_hdr.values(): + entry.addr = addr + addr += entry.entry_size + + def __init__(self): + self.compress_algo = None + self._gen_param() + + def image_crc(self, image): + crc = binascii.crc_hqx(image, 0) + crc = pack("; + + images { + kernel-1 { + description = "cvitek kernel"; + data = /incbin/("./Image.lzma"); + type = "kernel"; + arch = "arm64"; + os = "linux"; + compression = "lzma"; + load = <0x0 0x80200000>; + entry = <0x0 0x80200000>; + hash-2 { + algo = "crc32"; + }; + }; + + + /*FDT*/ + + fdt-sg2002_milkv_duo256m_arm_sd { + description = "cvitek device tree - sg2002_milkv_duo256m_arm_sd"; + data = /incbin/("./sg2002_milkv_duo256m_arm_sd.dtb"); + type = "flat_dt"; + arch = "arm64"; + compression = "none"; + hash-1 { + algo = "sha256"; + }; + }; + + + + }; + + /*CFG*/ + configurations { + + config-sg2002_milkv_duo256m_arm_sd { + description = "boot cvitek system with board sg2002_milkv_duo256m_arm_sd"; + kernel = "kernel-1"; + fdt = "fdt-sg2002_milkv_duo256m_arm_sd"; + }; + + }; + + +}; diff --git a/bsp/cvitek/cv18xx_aarch64/dtb/milkv-duo256m/sg2002_milkv_duo256m_arm_sd.dtb b/bsp/cvitek/cv18xx_aarch64/dtb/milkv-duo256m/sg2002_milkv_duo256m_arm_sd.dtb new file mode 100644 index 0000000000..6ef447d6e2 Binary files /dev/null and b/bsp/cvitek/cv18xx_aarch64/dtb/milkv-duo256m/sg2002_milkv_duo256m_arm_sd.dtb differ diff --git a/bsp/cvitek/cv18xx_aarch64/rtconfig.py b/bsp/cvitek/cv18xx_aarch64/rtconfig.py index c363513a18..54ea2201bb 100644 --- a/bsp/cvitek/cv18xx_aarch64/rtconfig.py +++ b/bsp/cvitek/cv18xx_aarch64/rtconfig.py @@ -45,3 +45,5 @@ if PLATFORM == 'gcc': DUMP_ACTION = OBJDUMP + ' -D -S $TARGET > rtt.asm\n' POST_ACTION = OBJCPY + ' -O binary $TARGET rtthread.bin\n' + SIZE + ' $TARGET \n' +POST_ACTION += OBJCPY + ' -O binary $TARGET Image \n' + SIZE + ' $TARGET \n' +POST_ACTION += 'cd .. && bash mksdimg.sh ' + os.getcwd() + ' Image \n'