/* * \brief Exynos5-specific implementation of the Block::Driver interface * \author Sebastian Sumpf * \date 2013-03-22 */ /* * Copyright (C) 2013 Genode Labs GmbH * * This file is part of the Genode OS framework, which is distributed * under the terms of the GNU General Public License version 2. */ #ifndef _DRIVER_H_ #define _DRIVER_H_ #include #include #include #include #include /* local includes */ #include namespace Block { using namespace Genode; class Exynos5_driver; } class Block::Exynos5_driver : public Block::Driver { private: struct Timer_delayer : Timer::Connection, Mmio::Delayer { /** * Implementation of 'Delayer' interface */ void usleep(unsigned us) { /* polling */ if (us == 0) return; Timer::Connection::usleep(us); } } _delayer; enum { MSH_BASE = 0x12200000, /* host controller base for eMMC */ MSH_SIZE = 0x10000, }; /* display sub system registers */ Attached_io_mem_dataspace _mmio; /* mobile storage host controller instance */ Exynos5_msh_controller _controller; bool const _use_dma; public: Exynos5_driver(bool use_dma) : _mmio(MSH_BASE, MSH_SIZE), _controller((addr_t)_mmio.local_addr(), _delayer, use_dma), _use_dma(use_dma) { Sd_card::Card_info const card_info = _controller.card_info(); PLOG("SD/MMC card detected"); PLOG("capacity: %zd MiB", card_info.capacity_mb()); } /***************************** ** Block::Driver interface ** *****************************/ Genode::size_t block_size() { return 512; } virtual Genode::size_t block_count() { return _controller.card_info().capacity_mb() * 1024 * 2; } void read(Genode::size_t block_number, Genode::size_t block_count, char *out_buffer) { if (!_controller.read_blocks(block_number, block_count, out_buffer)) throw Io_error(); } void write(Genode::size_t block_number, Genode::size_t block_count, char const *buffer) { if (!_controller.write_blocks(block_number, block_count, buffer)) throw Io_error(); } void read_dma(Genode::size_t block_number, Genode::size_t block_count, Genode::addr_t phys) { if (!_controller.read_blocks_dma(block_number, block_count, phys)) throw Io_error(); } void write_dma(Genode::size_t block_number, Genode::size_t block_count, Genode::addr_t phys) { if (!_controller.write_blocks_dma(block_number, block_count, phys)) throw Io_error(); } bool dma_enabled() { return _use_dma; } Ram_dataspace_capability alloc_dma_buffer(size_t size) { return Genode::env()->ram_session()->alloc(size, false); } void sync() {} }; #endif /* _DRIVER_H_ */