/* * \brief GPT utils * \author Josef Soentgen * \date 2018-05-01 */ /* * Copyright (C) 2018 Genode Labs GmbH * * This file is part of the Genode OS framework, which is distributed * under the terms of the GNU Affero General Public License version 3. */ #ifndef _UTIL_H_ #define _UTIL_H_ /* Genode includes */ #include #include #include #include namespace Util { using namespace Genode; void init_random(Genode::Allocator &); void get_random(uint8_t *dest, size_t len); using Label = Genode::String<128>; using Size_string = Genode::String<64>; uint64_t convert(Size_string const &); using sector_t = Block::sector_t; sector_t align_start(size_t, size_t, sector_t); sector_t size_to_lba(size_t, uint64_t); struct Block_io; uint32_t crc32(void const * const, size_t); size_t extract_ascii(char *, size_t, uint16_t const *, size_t); size_t convert_ascii(uint16_t *, size_t, uint8_t const *, size_t); /* * Wrapper to get suffixed uint64_t values */ class Number_of_bytes { uint64_t _n; public: /** * Default constructor */ Number_of_bytes() : _n(0) { } /** * Constructor, to be used implicitly via assignment operator */ Number_of_bytes(Genode::uint64_t n) : _n(n) { } /** * Convert number of bytes to 'size_t' value */ operator Genode::uint64_t() const { return _n; } void print(Output &output) const { using Genode::print; enum { KB = 1024UL, MB = KB*1024UL, GB = MB*1024UL }; if (_n == 0) print(output, 0); else if (_n % GB == 0) print(output, _n/GB, "G"); else if (_n % MB == 0) print(output, _n/MB, "M"); else if (_n % KB == 0) print(output, _n/KB, "K"); else print(output, _n); } }; inline size_t ascii_to(const char *s, Number_of_bytes &result) { unsigned long res = 0; /* convert numeric part of string */ int i = ascii_to_unsigned(s, res, 0); /* handle suffixes */ if (i > 0) switch (s[i]) { case 'G': res *= 1024; [[fallthrough]]; case 'M': res *= 1024; [[fallthrough]]; case 'K': res *= 1024; i++; default: break; } result = res; return i; } }; /* * Block_io wraps a Block::Connection for synchronous operations */ struct Util::Block_io { struct Io_error : Genode::Exception { }; using Packet_descriptor = Block::Packet_descriptor; Block::Connection<> &_block; Packet_descriptor _p; /** * Constructor * * \param block reference to underlying Block::Connection * \param block_size logical block size of the Block::Connection * \param lba LBA to start access from * \param count number of LBAs to access * \param write set type of operation, write if true, read * if false * * \throw Io_error */ Block_io(Block::Connection<> &block, size_t block_size, sector_t lba, size_t count, bool write = false, void const *data = nullptr, size_t len = 0) : _block(block), _p(_block.tx()->alloc_packet(block_size * count), write ? Packet_descriptor::WRITE : Packet_descriptor::READ, lba, count) { if (write) { if (data && len) { void *p = addr(); Genode::memcpy(p, data, len); } else { Genode::error("invalid data for write"); throw Io_error(); } } _block.tx()->submit_packet(_p); _p = _block.tx()->get_acked_packet(); if (!_p.succeeded()) { Genode::error("could not ", write ? "write" : "read", " block-range [", _p.block_number(), ",", _p.block_number() + count, ")"); _block.tx()->release_packet(_p); throw Io_error(); } } ~Block_io() { _block.tx()->release_packet(_p); } template T addr() { return reinterpret_cast(_block.tx()->packet_content(_p)); } }; #endif /* _UTIL_H_ */