b785ef9ed7
* memory setup using memblock * map pages later
166 lines
4.2 KiB
C
166 lines
4.2 KiB
C
/*
|
|
* Copyright (c) 2006-2023, RT-Thread Development Team
|
|
*
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
*
|
|
* Change Logs:
|
|
* Date Author Notes
|
|
* 2023-02-21 GuEe-GUI first version
|
|
*/
|
|
|
|
#include <rtthread.h>
|
|
#include <rtdevice.h>
|
|
|
|
#define DBG_TAG "rtdm.mnt"
|
|
#define DBG_LVL DBG_INFO
|
|
#include <rtdbg.h>
|
|
|
|
#include <stdlib.h>
|
|
#include <dfs_fs.h>
|
|
#ifdef RT_USING_FINSH
|
|
#include <msh.h>
|
|
#endif
|
|
#include <ioremap.h>
|
|
#include <mm_memblock.h>
|
|
|
|
#ifdef RT_USING_OFW
|
|
#define bootargs_select rt_ofw_bootargs_select
|
|
#else
|
|
#error Platform have not kernel parameters select interfaces!
|
|
#endif
|
|
|
|
static int rootfs_mnt_init(void)
|
|
{
|
|
rt_err_t err = -RT_ERROR;
|
|
void *fsdata = RT_NULL;
|
|
const char *cromfs_type = "crom";
|
|
const char *dev = bootargs_select("root=", 0);
|
|
const char *fstype = bootargs_select("rootfstype=", 0);
|
|
const char *rw = bootargs_select("rw", 0);
|
|
|
|
if (!dev || !fstype)
|
|
{
|
|
const char *name = "initrd";
|
|
rt_uint64_t initrd_start = 0, initrd_end = 0;
|
|
struct rt_mmblk_reg *iter = RT_NULL;
|
|
|
|
rt_slist_for_each_entry(iter, &(rt_memblock_get_reserved()->reg_list), node)
|
|
{
|
|
if (rt_strcmp(iter->memreg.name, name) == 0)
|
|
{
|
|
initrd_start = iter->memreg.start;
|
|
initrd_end = iter->memreg.end;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (initrd_start && initrd_end)
|
|
{
|
|
size_t initrd_size = initrd_end - initrd_start;
|
|
|
|
if ((fsdata = rt_ioremap_cached((void *)initrd_start, initrd_size)))
|
|
{
|
|
fstype = cromfs_type;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (fstype != cromfs_type && dev)
|
|
{
|
|
rt_tick_t timeout = 0;
|
|
const char *rootwait, *rootdelay = RT_NULL;
|
|
|
|
rootwait = bootargs_select("rootwait", 0);
|
|
|
|
/* Maybe it is undefined or 'rootwaitABC' */
|
|
if (!rootwait || *rootwait)
|
|
{
|
|
rootdelay = bootargs_select("rootdelay=", 0);
|
|
|
|
if (rootdelay)
|
|
{
|
|
timeout = rt_tick_from_millisecond(atoi(rootdelay));
|
|
}
|
|
|
|
rootwait = RT_NULL;
|
|
}
|
|
|
|
/*
|
|
* Delays in boot flow is a terrible behavior in RTOS, but the RT-Thread
|
|
* SDIO framework init the devices in a task that we need to wait for
|
|
* SDIO devices to init complete...
|
|
*
|
|
* WHAT THE F*CK PROBLEMS WILL HAPPENED?
|
|
*
|
|
* Your main PE, applications, services that depend on the root FS and
|
|
* the multi cores setup, init will delay, too...
|
|
*
|
|
* So, you can try to link this function to `INIT_APP_EXPORT` even later
|
|
* and remove the delays if you want to optimize the boot time and mount
|
|
* the FS auto.
|
|
*/
|
|
for (; rootdelay || rootwait; --timeout)
|
|
{
|
|
if (!rootwait && timeout == 0)
|
|
{
|
|
LOG_E("Wait for /dev/%s init time out", dev);
|
|
|
|
/*
|
|
* We don't return at once because the device driver may init OK
|
|
* when we break from this point, might as well give it another
|
|
* try.
|
|
*/
|
|
break;
|
|
}
|
|
|
|
if (rt_device_find(dev))
|
|
{
|
|
break;
|
|
}
|
|
|
|
rt_thread_mdelay(1);
|
|
}
|
|
}
|
|
|
|
if (fstype)
|
|
{
|
|
if (!(err = dfs_mount(dev, "/", fstype, rw ? 0 : ~0, fsdata)))
|
|
{
|
|
LOG_I("Mount root %s%s type=%s %s",
|
|
(dev && *dev) ? "on /dev/" : "",
|
|
(dev && *dev) ? dev : "\b",
|
|
fstype, "done");
|
|
}
|
|
else
|
|
{
|
|
LOG_W("Mount root %s%s type=%s %s",
|
|
(dev && *dev) ? "on /dev/" : "",
|
|
(dev && *dev) ? dev : "\b",
|
|
fstype, "fail");
|
|
|
|
if (fstype == cromfs_type)
|
|
{
|
|
rt_iounmap(fsdata);
|
|
}
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
INIT_ENV_EXPORT(rootfs_mnt_init);
|
|
|
|
static int fstab_mnt_init(void)
|
|
{
|
|
mkdir("/mnt", 0755);
|
|
|
|
#ifdef RT_USING_FINSH
|
|
/* Try mount by table */
|
|
msh_exec_script("fstab.sh", 16);
|
|
#endif
|
|
|
|
LOG_I("File system initialization done");
|
|
|
|
return 0;
|
|
}
|
|
INIT_FS_EXPORT(fstab_mnt_init);
|