/*
 * Copyright (c) Atmosphère-NX
 *
 * This program is free software; you can redistribute it and/or modify it
 * under the terms and conditions of the GNU General Public License,
 * version 2, as published by the Free Software Foundation.
 *
 * This program is distributed in the hope it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 * more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see .
 */
#include 
#include "os_resource_manager.hpp"
extern "C" { extern u8 __argdata__[]; }
namespace ams::os {
    namespace {
        class MemoryArranger {
            private:
                uintptr_t m_address;
            public:
                constexpr MemoryArranger(uintptr_t address) : m_address(address) { /* ... */ }
                template
                T *Arrange() {
                    this->Align(alignof(T));
                    return static_cast(this->Arrange(sizeof(T)));
                }
                void *Arrange(size_t size) {
                    const auto address = m_address;
                    m_address += size;
                    return reinterpret_cast(address);
                }
                void Align(size_t align) {
                    m_address = util::AlignUp(m_address, align);
                }
                char *ArrangeCharArray(size_t size) {
                    return reinterpret_cast(Arrange(size));
                }
        };
        bool HasArguments(uintptr_t args_region) {
            /* Check that the arguments region is read-write. */
            svc::MemoryInfo mi;
            svc::PageInfo pi;
            if (R_FAILED(svc::QueryMemory(std::addressof(mi), std::addressof(pi), args_region))) {
                return false;
            }
            return mi.permission == svc::MemoryPermission_ReadWrite;
        }
        const char *SkipSpace(const char *p, const char *end) {
            while (p < end && std::isspace(*p)) { ++p; }
            return p;
        }
        const char *GetTokenEnd(const char *p, const char *end) {
            while (p < end && !std::isspace(*p)) { ++p; }
            return p;
        }
        const char *GetQuotedTokenEnd(const char *p, const char *end) {
            while (p < end && *p != '"') { ++p; }
            return p;
        }
        int MakeArgv(char **out_argv_buf, char *arg_buf, const char *cmd_line, size_t cmd_line_size, int arg_max) {
            /* Prepare to parse arguments. */
                  auto idx = 0;
                  auto src = cmd_line;
                  auto dst = arg_buf;
            const auto end = src + cmd_line_size;
            /* Parse all tokens. */
            while (true) {
                /* Advance past any spaces. */
                src = SkipSpace(src, end);
                if (src >= end) {
                    break;
                }
                /* Check that we don't have too many arguments. */
                if (idx >= arg_max) {
                    break;
                }
                /* Find the start/end of the current argument token. */
                const char *arg_end;
                const char *src_next;
                if (*src == '"') {
                    ++src;
                    arg_end  = GetQuotedTokenEnd(src, end);
                    src_next = arg_end + 1;
                } else {
                    arg_end  = GetTokenEnd(src, end);
                    src_next = arg_end;
                }
                /* Determine token size. */
                const auto arg_size = arg_end - src;
                /* Set the argv pointer. */
                out_argv_buf[idx++] = dst;
                /* Copy the argument. */
                std::memcpy(dst, src, arg_size);
                dst += arg_size;
                /* Null-terminate the argument token. */
                *(dst++) = '\x00';
                /* Advance to next token. */
                src = src_next;
            }
            /* Null terminate the final token. */
            *(dst++) = '\x00';
            /* Null terminate argv. */
            out_argv_buf[idx] = nullptr;
            return idx;
        }
    }
    void SetHostArgc(int argc);
    void SetHostArgv(char **argv);
    void Initialize() {
        /* Only allow os::Initialize to be called once. */
        AMS_FUNCTION_LOCAL_STATIC_CONSTINIT(bool, s_initialized, false);
        if (s_initialized) {
            return;
        }
        s_initialized = true;
        /* Initialize the global os resource manager. */
        os::impl::ResourceManagerHolder::InitializeResourceManagerInstance();
        /* Setup host argc/argv as needed. */
        const uintptr_t args_region = reinterpret_cast(__argdata__);
        if (HasArguments(args_region)) {
            /* Create arguments memory arranger. */
            MemoryArranger arranger(args_region);
            /* Arrange. */
            const auto &header   = *arranger.Arrange();
            const char *cmd_line =  arranger.ArrangeCharArray(header.arguments_size);
            char *arg_buf        =  arranger.ArrangeCharArray(header.arguments_size + 2);
            char **argv_buf      =  arranger.Arrange();
            /* Determine extents. */
            const auto arg_buf_size = reinterpret_cast(argv_buf) - args_region;
            const auto arg_max      = (header.allocated_size - arg_buf_size) / sizeof(char *);
            /* Make argv. */
            const auto arg_count = MakeArgv(argv_buf, arg_buf, cmd_line, header.arguments_size, arg_max);
            /* Set host argc/argv. */
            os::SetHostArgc(arg_count);
            os::SetHostArgv(argv_buf);
        }
    }
}