/** * @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 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(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