/*
 * 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 <http://www.gnu.org/licenses/>.
 */
#if defined(ATMOSPHERE_IS_STRATOSPHERE)
#include <stratosphere.hpp>
#elif defined(ATMOSPHERE_IS_MESOSPHERE)
#include <mesosphere.hpp>
#elif defined(ATMOSPHERE_IS_EXOSPHERE)
#include <exosphere.hpp>
#else
#include <vapours.hpp>
#endif
#include "sdmmc_base_device_accessor.hpp"

namespace ams::sdmmc::impl {

    #if defined(AMS_SDMMC_THREAD_SAFE)

        #define AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX() std::scoped_lock lk(m_base_device->m_device_mutex)

    #else

        #define AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX()

    #endif

    void BaseDevice::GetLegacyCapacityParameters(u8 *out_c_size_mult, u8 *out_read_bl_len) const {
        AMS_ABORT_UNLESS(out_c_size_mult  != nullptr);
        AMS_ABORT_UNLESS(out_read_bl_len != nullptr);

        /* Extract C_SIZE_MULT and READ_BL_LEN from the CSD. */
        *out_c_size_mult = static_cast<u8>((m_csd[2] >> 7) & 0x7);
        *out_read_bl_len = static_cast<u8>((m_csd[4] >> 8) & 0xF);
    }

    Result BaseDevice::SetLegacyMemoryCapacity() {
        /* Get csize from the csd. */
        const u32 c_size = ((m_csd[3] >> 6) & 0x3FF) | ((m_csd[4] & 0x3) << 10);

        /* Get c_size_mult and read_bl_len. */
        u8 c_size_mult, read_bl_len;
        this->GetLegacyCapacityParameters(std::addressof(c_size_mult), std::addressof(read_bl_len));

        /* Validate the parameters. */
        R_UNLESS((read_bl_len + c_size_mult + 2) >= 9, sdmmc::ResultUnexpectedDeviceCsdValue());

        /* Set memory capacity. */
        m_memory_capacity          = (c_size + 1) << ((read_bl_len + c_size_mult + 2) - 9);
        m_is_valid_memory_capacity = true;

        R_SUCCEED();
    }

    Result BaseDevice::CheckDeviceStatus(u32 r1_resp) const {
        /* Check if there are any errors at all. */
        R_SUCCEED_IF((r1_resp & DeviceStatus_ErrorMask) == 0);

        /* Check errors individually. */
        #define AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(__ERROR__) R_UNLESS((r1_resp & DeviceStatus_##__ERROR__) == 0, sdmmc::ResultDeviceStatus##__ERROR__())

        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(ComCrcError);
        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(DeviceEccFailed);
        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(CcError);
        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(Error);

        if (this->GetDeviceType() == DeviceType_Mmc) {
            AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(SwitchError);
        }

        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(AddressMisaligned);
        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(BlockLenError);
        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(EraseSeqError);
        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(EraseParam);
        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(WpViolation);
        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(LockUnlockFailed);
        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(CidCsdOverwrite);
        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(WpEraseSkip);
        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(WpEraseSkip);
        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(EraseReset);
        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(IllegalCommand);
        AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR(AddressOutOfRange);

        #undef AMS_SDMMC_CHECK_DEVICE_STATUS_ERROR

        R_SUCCEED();
    }

    DeviceState BaseDevice::GetDeviceState(u32 r1_resp) const {
        return static_cast<DeviceState>((r1_resp & DeviceStatus_CurrentStateMask) >> DeviceStatus_CurrentStateShift);
    }

    Result BaseDeviceAccessor::IssueCommandAndCheckR1(u32 *out_response, u32 command_index, u32 command_arg, bool is_busy, DeviceState expected_state, u32 status_ignore_mask) const {
        /* Issue the command. */
        constexpr ResponseType CommandResponseType = ResponseType_R1;
        Command command(command_index, command_arg, CommandResponseType, is_busy);
        R_TRY(m_host_controller->IssueCommand(std::addressof(command)));

        /* Get the response. */
        AMS_ABORT_UNLESS(out_response != nullptr);
        m_host_controller->GetLastResponse(out_response, sizeof(u32), CommandResponseType);

        /* Mask out the ignored status bits. */
        if (status_ignore_mask != 0) {
            *out_response &= ~status_ignore_mask;
        }

        /* Check the r1 response for errors. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        R_TRY(m_base_device->CheckDeviceStatus(*out_response));

        /* Check the device state. */
        if (expected_state != DeviceState_Unknown) {
            R_UNLESS(m_base_device->GetDeviceState(*out_response) == expected_state, sdmmc::ResultUnexpectedDeviceState());
        }

        R_SUCCEED();
    }

    Result BaseDeviceAccessor::IssueCommandGoIdleState() const {
        /* Issue the command. */
        Command command(CommandIndex_GoIdleState, 0, ResponseType_R0, false);
        R_RETURN(m_host_controller->IssueCommand(std::addressof(command)));
    }

    Result BaseDeviceAccessor::IssueCommandAllSendCid(void *dst, size_t dst_size) const {
        /* Issue the command. */
        constexpr ResponseType CommandResponseType = ResponseType_R2;
        Command command(CommandIndex_AllSendCid, 0, CommandResponseType, false);
        R_TRY(m_host_controller->IssueCommand(std::addressof(command)));

        /* Copy the data out. */
        AMS_ABORT_UNLESS(dst != nullptr);
        AMS_ABORT_UNLESS(util::IsAligned(reinterpret_cast<uintptr_t>(dst), alignof(u32)));
        AMS_ABORT_UNLESS(dst_size >= DeviceCidSize);
        m_host_controller->GetLastResponse(static_cast<u32 *>(dst), DeviceCidSize, CommandResponseType);

        R_SUCCEED();
    }

    Result BaseDeviceAccessor::IssueCommandSelectCard() const {
        /* Get the command argument. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        const u32 arg = static_cast<u32>(m_base_device->GetRca()) << 16;

        /* Issue the command. */
        R_RETURN(this->IssueCommandAndCheckR1(CommandIndex_SelectCard, arg, true, DeviceState_Unknown));
    }

    Result BaseDeviceAccessor::IssueCommandSendCsd(void *dst, size_t dst_size) const {
        /* Get the command argument. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        const u32 arg = static_cast<u32>(m_base_device->GetRca()) << 16;

        /* Issue the command. */
        constexpr ResponseType CommandResponseType = ResponseType_R2;
        Command command(CommandIndex_SendCsd, arg, CommandResponseType, false);
        R_TRY(m_host_controller->IssueCommand(std::addressof(command)));

        /* Copy the data out. */
        AMS_ABORT_UNLESS(dst != nullptr);
        AMS_ABORT_UNLESS(util::IsAligned(reinterpret_cast<uintptr_t>(dst), alignof(u32)));
        AMS_ABORT_UNLESS(dst_size >= DeviceCsdSize);
        m_host_controller->GetLastResponse(static_cast<u32 *>(dst), DeviceCsdSize, CommandResponseType);

        R_SUCCEED();
    }

    Result BaseDeviceAccessor::IssueCommandSendStatus(u32 *out_device_status, u32 status_ignore_mask) const {
        /* Get the command argument. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        const u32 arg = static_cast<u32>(m_base_device->GetRca()) << 16;

        /* Issue the command. */
        R_RETURN(this->IssueCommandAndCheckR1(out_device_status, CommandIndex_SendStatus, arg, false, DeviceState_Tran, status_ignore_mask));
    }

    Result BaseDeviceAccessor::IssueCommandSetBlockLenToSectorSize() const {
        /* Issue the command. */
        R_RETURN(this->IssueCommandAndCheckR1(CommandIndex_SetBlockLen, SectorSize, false, DeviceState_Tran));
    }

    Result BaseDeviceAccessor::IssueCommandMultipleBlock(u32 *out_num_transferred_blocks, u32 sector_index, u32 num_sectors, void *buf, bool is_read) const {
        /* Get the argument. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        const u32 arg = m_base_device->IsHighCapacity() ? sector_index : sector_index * SectorSize;

        /* Get the command index and transfer direction. */
        const u32 command_index   = is_read ? CommandIndex_ReadMultipleBlock   : CommandIndex_WriteMultipleBlock;
        const auto xfer_direction = is_read ? TransferDirection_ReadFromDevice : TransferDirection_WriteToDevice;

        /* Issue the command. */
        constexpr ResponseType CommandResponseType = ResponseType_R1;
        Command command(command_index, arg, CommandResponseType, false);
        TransferData xfer_data(buf, SectorSize, num_sectors, xfer_direction, true, true);
        Result result = m_host_controller->IssueCommand(std::addressof(command), std::addressof(xfer_data), out_num_transferred_blocks);

        /* Handle the failure case. */
        if (R_FAILED(result)) {
            /* Check if we were removed. */
            R_TRY(this->CheckRemoved());

            /* By default, we'll want to return the result we just got. */
            Result result_to_return = result;

            /* Stop transmission. */
            u32 resp = 0;
            result = m_host_controller->IssueStopTransmissionCommand(std::addressof(resp));
            if (R_SUCCEEDED(result)) {
                result = m_base_device->CheckDeviceStatus(resp & (~DeviceStatus_IllegalCommand));
                if (R_FAILED(result)) {
                    result_to_return = result;
                }
            }

            /* Check if we were removed. */
            R_TRY(this->CheckRemoved());

            /* Get the device status. */
            u32 device_status = 0;
            result = this->IssueCommandSendStatus(std::addressof(device_status), DeviceStatus_IllegalCommand);

            /* If there's a device status error we don't already have, we prefer to return it. */
            if (!sdmmc::ResultDeviceStatusHasError::Includes(result_to_return) && sdmmc::ResultDeviceStatusHasError::Includes(result)) {
                result_to_return = result;
            }

            /* Return the result we chose. */
            R_RETURN(result_to_return);
        }

        /* Get the responses. */
        u32 resp, st_resp;
        m_host_controller->GetLastResponse(std::addressof(resp), sizeof(resp), CommandResponseType);
        m_host_controller->GetLastStopTransmissionResponse(std::addressof(st_resp), sizeof(st_resp));

        /* Check the device status. */
        R_TRY(m_base_device->CheckDeviceStatus(resp));

        /* Decide on what errors to ignore. */
        u32 status_ignore_mask = 0;
        if (is_read) {
            AMS_ABORT_UNLESS(out_num_transferred_blocks != nullptr);
            if ((*out_num_transferred_blocks + sector_index) == m_base_device->GetMemoryCapacity()) {
                status_ignore_mask = DeviceStatus_AddressOutOfRange;
            }
        }

        /* Check the device status. */
        R_TRY(m_base_device->CheckDeviceStatus(st_resp & ~status_ignore_mask));

        R_SUCCEED();
    }

    Result BaseDeviceAccessor::ReadWriteSingle(u32 *out_num_transferred_blocks, u32 sector_index, u32 num_sectors, void *buf, bool is_read) const {
        /* Issue the read/write command. */
        AMS_ABORT_UNLESS(out_num_transferred_blocks != nullptr);
        R_TRY(this->IssueCommandMultipleBlock(out_num_transferred_blocks, sector_index, num_sectors, buf, is_read));

        /* Decide on what errors to ignore. */
        u32 status_ignore_mask = 0;
        if (is_read) {
            AMS_ABORT_UNLESS(m_base_device != nullptr);
            if ((*out_num_transferred_blocks + sector_index) == m_base_device->GetMemoryCapacity()) {
                status_ignore_mask = DeviceStatus_AddressOutOfRange;
            }
        }

        /* Get and check the status. */
        u32 device_status;
        R_TRY(this->IssueCommandSendStatus(std::addressof(device_status), status_ignore_mask));

        R_SUCCEED();
    }

    Result BaseDeviceAccessor::ReadWriteMultiple(u32 sector_index, u32 num_sectors, u32 sector_index_alignment, void *buf, size_t buf_size, bool is_read) {
        /* Verify that we can send the command. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);

        /* If we want to read zero sectors, there's no work for us to do. */
        R_SUCCEED_IF(num_sectors == 0);

        /* Check that the buffer is big enough for the sectors we're reading. */
        AMS_ABORT_UNLESS((buf_size / SectorSize) >= num_sectors);

        /* Read sectors repeatedly until we've read all the ones we want. */
        u32 cur_sector_index  = sector_index;
        u32 remaining_sectors = num_sectors;
        u8 *cur_buf           = static_cast<u8 *>(buf);
        while (remaining_sectors > 0) {
            /* Determine how many sectors we can read in this iteration. */
            u32 cur_sectors = remaining_sectors;
            if (sector_index_alignment > 0) {
                AMS_ABORT_UNLESS((cur_sector_index % sector_index_alignment) == 0);

                const u32 max_sectors = m_host_controller->GetMaxTransferNumBlocks();
                if (remaining_sectors > max_sectors) {
                    cur_sectors = max_sectors - (max_sectors % sector_index_alignment);
                }
            }

            /* Try to perform the read/write. */
            u32 num_transferred_blocks = 0;
            Result result = this->ReadWriteSingle(std::addressof(num_transferred_blocks), cur_sector_index, cur_sectors, cur_buf, is_read);
            if (R_FAILED(result)) {
                /* Check if we were removed. */
                R_TRY(this->CheckRemoved());

                /* Log that we failed to read/write. */
                this->PushErrorLog(false, "%s %X %X:%X", is_read ? "R" : "W", cur_sector_index, cur_sectors, result.GetValue());

                /* Retry the read/write. */
                num_transferred_blocks = 0;
                result = this->ReadWriteSingle(std::addressof(num_transferred_blocks), cur_sector_index, cur_sectors, cur_buf, is_read);
                if (R_FAILED(result)) {
                    /* Check if we were removed. */
                    R_TRY(this->CheckRemoved());

                    /* Log that we failed to read/write. */
                    this->PushErrorLog(false, "%s %X %X:%X", is_read ? "R" : "W", cur_sector_index, cur_sectors, result.GetValue());

                    /* Re-startup the connection, to see if that helps. */
                    R_TRY(this->ReStartup());

                    /* Retry the read/write a third time. */
                    num_transferred_blocks = 0;
                    result = this->ReadWriteSingle(std::addressof(num_transferred_blocks), cur_sector_index, cur_sectors, cur_buf, is_read);
                    if (R_FAILED(result)) {
                        /* Log that we failed after a re-startup. */
                        this->PushErrorLog(true, "%s %X %X:%X", is_read ? "R" : "W", cur_sector_index, cur_sectors, result.GetValue());
                        R_RETURN(result);
                    }

                    /* Log that we succeeded after a retry. */
                    this->PushErrorLog(true, "%s %X %X:0", is_read ? "R" : "W", cur_sector_index, cur_sectors);

                    /* Increment the number of error corrections we've done. */
                    ++m_num_read_write_error_corrections;
                }
            }

            /* Update our tracking variables. */
            AMS_ABORT_UNLESS(remaining_sectors >= num_transferred_blocks);
            remaining_sectors -= num_transferred_blocks;
            cur_sector_index  += num_transferred_blocks;
            cur_buf           += num_transferred_blocks * SectorSize;
        }

        R_SUCCEED();
    }

    #if defined(AMS_SDMMC_USE_DEVICE_VIRTUAL_ADDRESS)
    void BaseDeviceAccessor::RegisterDeviceVirtualAddress(uintptr_t buffer, size_t buffer_size, ams::dd::DeviceVirtualAddress buffer_device_virtual_address) {
        /* Lock exclusive access of the base device. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX();

        /* Register the address. */
        return m_host_controller->RegisterDeviceVirtualAddress(buffer, buffer_size, buffer_device_virtual_address);
    }

    void BaseDeviceAccessor::UnregisterDeviceVirtualAddress(uintptr_t buffer, size_t buffer_size, ams::dd::DeviceVirtualAddress buffer_device_virtual_address) {
        /* Lock exclusive access of the base device. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX();

        /* Register the address. */
        return m_host_controller->UnregisterDeviceVirtualAddress(buffer, buffer_size, buffer_device_virtual_address);
    }
    #endif

    Result BaseDeviceAccessor::Activate() {
        /* Lock exclusive access of the base device. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX();

        /* Check that the device is awake. */
        R_UNLESS(m_base_device->IsAwake(), sdmmc::ResultNotAwakened());

        /* If the device is already active, we don't need to do anything. */
        R_SUCCEED_IF(m_base_device->IsActive());

        /* Activate the base device. */
        auto activate_guard = SCOPE_GUARD { ++m_num_activation_failures; };
        R_TRY(this->OnActivate());

        /* We successfully activated the device. */
        activate_guard.Cancel();
        m_base_device->SetActive();

        R_SUCCEED();
    }

    void BaseDeviceAccessor::Deactivate() {
        /* Lock exclusive access of the base device. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX();

        /* Deactivate the base device. */
        if (m_base_device->IsActive()) {
            m_host_controller->Shutdown();
            m_base_device->Deactivate();
        }
    }

    Result BaseDeviceAccessor::ReadWrite(u32 sector_index, u32 num_sectors, void *buffer, size_t buffer_size, bool is_read) {
        /* Lock exclusive access of the base device. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX();

        /* Check that the device is accessible. */
        R_TRY(m_base_device->CheckAccessible());

        /* Perform the read/write. */
        auto rw_guard = SCOPE_GUARD { ++m_num_read_write_failures; };
        R_TRY(this->OnReadWrite(sector_index, num_sectors, buffer, buffer_size, is_read));

        /* We successfully performed the read/write. */
        rw_guard.Cancel();
        R_SUCCEED();
    }

    Result BaseDeviceAccessor::CheckConnection(SpeedMode *out_speed_mode, BusWidth *out_bus_width) {
        /* Lock exclusive access of the base device. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX();

        /* Check that the device is accessible. */
        R_TRY(m_base_device->CheckAccessible());

        /* Get the current speed mode/bus width. */
        *out_speed_mode = m_host_controller->GetSpeedMode();
        *out_bus_width  = m_host_controller->GetBusWidth();

        /* Verify that we can get the status. */
        R_TRY(m_host_controller->GetInternalStatus());

        R_SUCCEED();
    }

    Result BaseDeviceAccessor::GetMemoryCapacity(u32 *out_sectors) const {
        /* Lock exclusive access of the base device. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX();

        /* Check that the device is accessible. */
        R_TRY(m_base_device->CheckAccessible());

        /* Get the capacity. */
        AMS_ABORT_UNLESS(out_sectors != nullptr);
        *out_sectors = m_base_device->GetMemoryCapacity();

        R_SUCCEED();
    }

    Result BaseDeviceAccessor::GetDeviceStatus(u32 *out) const {
        /* Lock exclusive access of the base device. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX();

        /* Check that the device is accessible. */
        R_TRY(m_base_device->CheckAccessible());

        /* Get the status. */
        R_TRY(this->IssueCommandSendStatus(out, 0));

        R_SUCCEED();
    }

    Result BaseDeviceAccessor::GetOcr(u32 *out) const {
        /* Lock exclusive access of the base device. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX();

        /* Check that the device is accessible. */
        R_TRY(m_base_device->CheckAccessible());

        /* Get the ocr. */
        AMS_ABORT_UNLESS(out != nullptr);
        *out = m_base_device->GetOcr();

        R_SUCCEED();
    }

    Result BaseDeviceAccessor::GetRca(u16 *out) const {
        /* Lock exclusive access of the base device. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX();

        /* Check that the device is accessible. */
        R_TRY(m_base_device->CheckAccessible());

        /* Get the rca. */
        AMS_ABORT_UNLESS(out != nullptr);
        *out = m_base_device->GetRca();

        R_SUCCEED();
    }

    Result BaseDeviceAccessor::GetCid(void *out, size_t size) const {
        /* Lock exclusive access of the base device. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX();

        /* Check that the device is accessible. */
        R_TRY(m_base_device->CheckAccessible());

        /* Get the cid. */
        m_base_device->GetCid(out, size);

        R_SUCCEED();
    }

    Result BaseDeviceAccessor::GetCsd(void *out, size_t size) const {
        /* Lock exclusive access of the base device. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX();

        /* Check that the device is accessible. */
        R_TRY(m_base_device->CheckAccessible());

        /* Get the csd. */
        m_base_device->GetCsd(out, size);

        R_SUCCEED();
    }

    void BaseDeviceAccessor::GetAndClearErrorInfo(ErrorInfo *out_error_info, size_t *out_log_size, char *out_log_buffer, size_t log_buffer_size) {
        /* Lock exclusive access of the base device. */
        AMS_ABORT_UNLESS(m_base_device != nullptr);
        AMS_SDMMC_LOCK_BASE_DEVICE_MUTEX();

        /* Set the output error info. */
        AMS_ABORT_UNLESS(out_error_info != nullptr);
        out_error_info->num_activation_failures           = m_num_activation_failures;
        out_error_info->num_activation_error_corrections  = m_num_activation_error_corrections;
        out_error_info->num_read_write_failures           = m_num_read_write_failures;
        out_error_info->num_read_write_error_corrections  = m_num_read_write_error_corrections;
        this->ClearErrorInfo();

        /* Check if we should write logs. */
        if (out_log_size == nullptr) {
            return;
        }

        /* Check if we can write logs. */
        if (out_log_buffer == nullptr || log_buffer_size == 0) {
            *out_log_size = 0;
            return;
        }

        /* Get and clear our logs. */
        #if defined(AMS_SDMMC_USE_LOGGER)
        {
            if (m_error_logger.HasLog()) {
                this->PushErrorTimeStamp();

                *out_log_size = m_error_logger.GetAndClearLogs(out_log_buffer, log_buffer_size);
            } else {
                *out_log_size = 0;
            }
        }
        #else
        {
            *out_log_size = 0;
        }
        #endif
    }

}