/*
 * 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 "updater_files.hpp"
namespace ams::updater {
    Result ReadFile(size_t *out_size, void *dst, size_t dst_size, const char *path) {
        /* Open the file. */
        fs::FileHandle file;
        R_TRY_CATCH(fs::OpenFile(std::addressof(file), path, fs::OpenMode_Read)) {
            R_CONVERT(fs::ResultPathNotFound, updater::ResultInvalidBootImagePackage())
        } R_END_TRY_CATCH;
        ON_SCOPE_EXIT { fs::CloseFile(file); };
        std::memset(dst, 0, dst_size);
        R_RETURN(fs::ReadFile(out_size, file, 0, dst, dst_size, fs::ReadOption()));
    }
    Result GetFileHash(size_t *out_size, void *dst_hash, const char *path, void *work_buffer, size_t work_buffer_size) {
        /* Open the file. */
        fs::FileHandle file;
        R_TRY_CATCH(fs::OpenFile(std::addressof(file), path, fs::OpenMode_Read)) {
            R_CONVERT(fs::ResultPathNotFound, updater::ResultInvalidBootImagePackage())
        } R_END_TRY_CATCH;
        ON_SCOPE_EXIT { fs::CloseFile(file); };
        /* Read in chunks, hashing as we go. */
        crypto::Sha256Generator generator;
        generator.Initialize();
        size_t total_size = 0;
        while (true) {
            size_t size;
            R_TRY(fs::ReadFile(std::addressof(size), file, total_size, work_buffer, work_buffer_size, fs::ReadOption()));
            generator.Update(work_buffer, size);
            total_size += size;
            if (size != work_buffer_size) {
                break;
            }
        }
        generator.GetHash(dst_hash, crypto::Sha256Generator::HashSize);
        *out_size = total_size;
        R_SUCCEED();
    }
}