/*
 * Copyright (c) 2018-2019 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 .
 */
#pragma once
#include 
#include 
class IDirectory {
    public:
        virtual ~IDirectory() {}
        Result Read(uint64_t *out_count, FsDirectoryEntry *out_entries, uint64_t max_entries) {
            if (out_count == nullptr) {
                return ResultFsNullptrArgument;
            }
            if (max_entries == 0) {
                *out_count = 0;
                return ResultSuccess;
            }
            if (out_entries == nullptr) {
                return ResultFsNullptrArgument;
            }
            return ReadImpl(out_count, out_entries, max_entries);
        }
        Result GetEntryCount(uint64_t *count) {
            if (count == nullptr) {
                return ResultFsNullptrArgument;
            }
            return GetEntryCountImpl(count);
        }
    protected:
        /* ...? */
    private:
        virtual Result ReadImpl(uint64_t *out_count, FsDirectoryEntry *out_entries, uint64_t max_entries) = 0;
        virtual Result GetEntryCountImpl(uint64_t *count) = 0;
};
class IDirectoryInterface : public IServiceObject {
    private:
        enum class CommandId {
            Read          = 0,
            GetEntryCount = 1,
        };
    private:
        std::unique_ptr base_dir;
    public:
        IDirectoryInterface(IDirectory *d) : base_dir(d) {
            /* ... */
        };
        IDirectoryInterface(std::unique_ptr d) : base_dir(std::move(d)) {
            /* ... */
        };
    private:
        /* Actual command API. */
        virtual Result Read(OutBuffer buffer, Out out_count) final {
            return this->base_dir->Read(out_count.GetPointer(), buffer.buffer, buffer.num_elements);
        };
        virtual Result GetEntryCount(Out out_count) final {
           return this->base_dir->GetEntryCount(out_count.GetPointer());
        };
    public:
        DEFINE_SERVICE_DISPATCH_TABLE {
            /* 1.0.0- */
            MAKE_SERVICE_COMMAND_META(IDirectoryInterface, Read),
            MAKE_SERVICE_COMMAND_META(IDirectoryInterface, GetEntryCount),
        };
};
class ProxyDirectory : public IDirectory {
    private:
        std::unique_ptr base_dir;
    public:
        ProxyDirectory(FsDir *d) : base_dir(d) {
            /* ... */
        }
        ProxyDirectory(std::unique_ptr d) : base_dir(std::move(d)) {
            /* ... */
        }
        ProxyDirectory(FsDir d) {
            this->base_dir = std::make_unique(d);
        }
        virtual ~ProxyDirectory() {
            fsDirClose(this->base_dir.get());
        }
    public:
        virtual Result ReadImpl(uint64_t *out_count, FsDirectoryEntry *out_entries, uint64_t max_entries) {
            size_t count;
            R_TRY(fsDirRead(this->base_dir.get(), 0, &count, max_entries, out_entries));
            *out_count = count;
            return ResultSuccess;
        }
        virtual Result GetEntryCountImpl(uint64_t *count) {
            return fsDirGetEntryCount(this->base_dir.get(), count);
        }
};