rusefi-full/firmware/hw_layer/mass_storage/null_device.cpp

127 lines
3.3 KiB
C++

/**
* @file zero_device.cpp
*
* @date Feb 12, 2021
* @author Matthew Kennedy, (c) 2021
*
* This file implements a special block device that simply reports "no media"
* Use it when you need to mount *something* but don't have an SD card available.
*/
#include "hal.h"
#if EFI_EMBED_INI_MSD
#include "ramdisk.h"
#include "compressed_block_device.h"
#ifdef EFI_USE_COMPRESSED_INI_MSD
#include "ramdisk_image_compressed.h"
#else
#include "ramdisk_image.h"
#endif
// If the ramdisk image told us not to use it, don't use it.
#ifdef RAMDISK_INVALID
#undef EFI_EMBED_INI_MSD
#define EFI_EMBED_INI_MSD FALSE
#endif
#endif
#include <cstring>
struct NullDevice {
const struct BaseBlockDeviceVMT *vmt;
_base_block_device_data
};
static bool nd_is_inserted(void*) {
// This function is the whole point - we have no media!
return false;
}
static bool nd_is_protected(void*) {
return false;
}
static bool nd_return_success(void*) {
return HAL_SUCCESS;
}
static bool nd_return_success_read(void*, uint32_t, uint8_t* buffer, uint32_t n) {
// write zeroes to the buffer to prevent somebody reading random memory
memset(buffer, 0, n);
return HAL_SUCCESS;
}
static bool nd_return_success_write(void*, uint32_t, const uint8_t*, uint32_t) {
return HAL_SUCCESS;
}
static bool nd_get_info(void*, BlockDeviceInfo* bdip) {
// We have to report non-zero size here because Windows
// will query the size of the block device even if we indicate
// that the device has no media
// If we report zeroes, it breaks USB until you unplug this device
bdip->blk_num = 1000;
bdip->blk_size = 512;
return HAL_SUCCESS;
}
static const struct BaseBlockDeviceVMT ndVmt = {
(size_t)0, // instanceOffset
nd_is_inserted,
nd_is_protected,
// These functions just claim success to make the host happy
nd_return_success,
nd_return_success,
nd_return_success_read,
nd_return_success_write,
nd_return_success,
nd_get_info
};
#if EFI_EMBED_INI_MSD
#ifdef EFI_USE_COMPRESSED_INI_MSD
static CompressedBlockDevice cbd;
#else
static RamDisk ramdisk;
#endif
#else
// This device is always ready and has no state
static NullDevice nd = { &ndVmt, BLK_READY };
#endif
#if HAL_USE_USB_MSD
void msdMountNullDevice(USBMassStorageDriver* msdp, USBDriver *usbp, uint8_t* blkbuf, const scsi_inquiry_response_t* inquiry) {
// TODO: implement multi-LUN so we can mount the ini image and SD card at the same time
#if EFI_EMBED_INI_MSD
#ifdef EFI_USE_COMPRESSED_INI_MSD
uzlib_init();
compressedBlockDeviceObjectInit(&cbd);
compressedBlockDeviceStart(&cbd, ramdisk_image_gz, sizeof(ramdisk_image_gz));
msdStart(msdp, usbp, (BaseBlockDevice*)&cbd, blkbuf, inquiry, nullptr);
#else // not EFI_USE_COMPRESSED_INI_MSD
ramdiskObjectInit(&ramdisk);
constexpr size_t ramdiskSize = sizeof(ramdisk_image);
constexpr size_t blockSize = 512;
constexpr size_t blockCount = ramdiskSize / blockSize;
// Ramdisk should be a round number of blocks
static_assert(ramdiskSize % blockSize == 0);
ramdiskStart(&ramdisk, const_cast<uint8_t*>(ramdisk_image), blockSize, blockCount, /*readonly =*/ true);
msdStart(msdp, usbp, (BaseBlockDevice*)&ramdisk, blkbuf, inquiry, nullptr);
#endif // EFI_USE_COMPRESSED_INI_MSD
#else // not EFI_EMBED_INI_MSD
// No embedded ini file, just mount the null device instead
msdStart(msdp, usbp, (BaseBlockDevice*)&nd, blkbuf, inquiry, nullptr);
#endif
}
#endif