/* * \brief Stream of block-operation requests * \author Norman Feske * \date 2018-12-17 */ /* * 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 _INCLUDE__BLOCK__REQUEST_STREAM_H_ #define _INCLUDE__BLOCK__REQUEST_STREAM_H_ /* Genode includes */ #include #include #include namespace Block { struct Request_stream; } class Block::Request_stream : Genode::Noncopyable { public: /** * Interface for accessing the content of a 'Request' * * The 'Payload' is separate from 'Request_stream' to allow its use * as argument without exposing the entirely of the 'Request_stream' * to the called code. */ class Payload : Noncopyable { private: friend class Request_stream; Genode::addr_t const _base; Genode::size_t const _size; Genode::uint32_t const _block_size; /** * Return pointer to the first byte of the request content */ void *_request_ptr(Block::Request request) const { return (void *)(_base + request.offset); } /** * Return request size in bytes */ Genode::size_t _request_size(Block::Request const &request) const { return request.count * _block_size; } bool _valid_range(Block::Request const &request) const { /* local address of the last byte of the request */ Genode::addr_t const request_end = _base + request.offset + _request_size(request) - 1; /* check for integer overflow and zero-sized request */ if (request_end <= _base) return false; /* check for upper bound */ if (request_end > _base + _size - 1) return false; return true; } Payload(Genode::addr_t base, Genode::size_t size, Genode::uint32_t block_size) : _base(base), _size(size), _block_size(block_size) { } public: /** * Call functor 'fn' with the pointer and size of the 'request' * content as arguments * * If the request does not carry any payload, 'fn' is not * called. */ template void with_content(Block::Request request, FN const &fn) const { if (_valid_range(request)) fn(_request_ptr(request), _request_size(request)); } }; private: Packet_stream_tx::Rpc_object _tx; typedef Genode::Packet_stream_sink Tx_sink; Payload const _payload; public: Request_stream(Genode::Region_map &rm, Genode::Dataspace_capability ds, Genode::Entrypoint &ep, Genode::Signal_context_capability sigh, Genode::uint32_t block_size) : _tx(ds, rm, ep.rpc_ep()), _payload(_tx.sink()->ds_local_base(), _tx.sink()->ds_size(), block_size) { _tx.sigh_ready_to_ack(sigh); _tx.sigh_packet_avail(sigh); } virtual ~Request_stream() { _tx.sigh_ready_to_ack(Genode::Signal_context_capability()); _tx.sigh_packet_avail(Genode::Signal_context_capability()); } Genode::Capability tx_cap() { return _tx.cap(); } /** * Call functor 'fn' with 'Payload' interface as argument * * The 'Payload' interface allows the functor to access the content * of a request by calling 'Payload::with_content'. */ template void with_payload(FN const &fn) const { fn(_payload); } /** * Call functor 'fn' with the pointer and size to the 'request' content * * This is a wrapper for 'Payload::with_content'. It is convenient * in situations where the 'Payload' interface does not need to be * propagated as argument. */ template void with_content(Request const &request, FN const &fn) const { _payload.with_content(request, fn); } enum class Response { ACCEPTED, REJECTED, RETRY }; /** * Call functor 'fn' for each pending request, with its packet as argument * * The boolean return value of the functor expresses whether the request * was accepted or not. If it was accepted, the request is removed from the * packet stream. If the request could not be accepted, the iteration * aborts and the request packet stays in the packet stream. */ template void with_requests(FN const &fn) { Tx_sink &tx_sink = *_tx.sink(); using namespace Genode; for (;;) { if (!tx_sink.packet_avail()) return; typedef Block::Packet_descriptor Packet_descriptor; Packet_descriptor const packet = tx_sink.peek_packet(); auto operation = [] (Packet_descriptor::Opcode op) { switch (op) { case Packet_descriptor::READ: return Request::Operation::READ; case Packet_descriptor::WRITE: return Request::Operation::WRITE; case Packet_descriptor::END: return Request::Operation::INVALID; }; return Request::Operation::INVALID; }; bool const packet_valid = (tx_sink.packet_valid(packet) && (packet.offset() >= 0) && (packet.size() <= (size_t)((uint32_t)~0UL))); Request request { .operation = operation(packet.operation()), .success = Request::Success::FALSE, .block_number = (uint64_t)packet.block_number(), .offset = (uint64_t)packet.offset(), .count = (uint32_t)packet.block_count(), .tag = 0}; Response const response = packet_valid ? fn(request) : Response::REJECTED; bool progress = false; switch (response) { case Response::REJECTED: /* * Acknowledge rejected packet if there is enough room in * the acknowledgement queue. Otherwise, the rejected * packet stays in the request queue and is evaluated * again. */ if (tx_sink.ack_slots_free()) { (void)tx_sink.try_get_packet(); tx_sink.try_ack_packet(packet); progress = true; } break; case Response::ACCEPTED: (void)tx_sink.try_get_packet(); progress = true; break; case Response::RETRY: break; } /* * Stop iterating of no request-queue elements can be consumed. */ if (!progress) break; } } class Ack : Noncopyable { private: friend class Request_stream; Tx_sink &_tx_sink; bool _submitted = false; Genode::uint32_t const _block_size; Ack(Tx_sink &tx_sink, Genode::uint32_t block_size) : _tx_sink(tx_sink), _block_size(block_size) { } public: void submit(Block::Request request) { if (_submitted) { Genode::warning("attempt to ack the same packet twice"); return; } typedef Block::Packet_descriptor Packet_descriptor; Packet_descriptor packet { (Genode::off_t)request.offset, request.count * _block_size }; auto opcode = [] (Request::Operation operation) { switch (operation) { case Request::Operation::READ: return Packet_descriptor::READ; case Request::Operation::WRITE: return Packet_descriptor::WRITE; case Request::Operation::SYNC: return Packet_descriptor::END; case Request::Operation::INVALID: return Packet_descriptor::END; }; return Packet_descriptor::END; }; packet = Packet_descriptor(packet, opcode(request.operation), request.block_number, request.count); packet.succeeded(request.success == Request::Success::TRUE); _tx_sink.try_ack_packet(packet); _submitted = true; } }; /** * Try to submit acknowledgement packets * * The method repeatedly calls the functor 'fn' with an 'Ack' reference, * which provides an interface to 'submit' one acknowledgement. The * iteration stops when the acknowledgement queue is fully populated or if * the functor does not call 'Ack::submit'. */ template void try_acknowledge(FN const &fn) { Tx_sink &tx_sink = *_tx.sink(); while (tx_sink.ack_slots_free()) { Ack ack(tx_sink, _payload._block_size); fn(ack); if (!ack._submitted) break; } } void wakeup_client() { _tx.sink()->wakeup(); } }; #endif /* _INCLUDE__BLOCK__REQUEST_STREAM_H_ */