/*
 * 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);