2011-12-22 16:19:25 +01:00
|
|
|
/*
|
|
|
|
* \brief Block session interface.
|
|
|
|
* \author Stefan Kalkowski
|
|
|
|
* \date 2010-07-06
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
2017-02-20 13:23:52 +01:00
|
|
|
* Copyright (C) 2010-2017 Genode Labs GmbH
|
2011-12-22 16:19:25 +01:00
|
|
|
*
|
|
|
|
* This file is part of the Genode OS framework, which is distributed
|
2017-02-20 13:23:52 +01:00
|
|
|
* under the terms of the GNU Affero General Public License version 3.
|
2011-12-22 16:19:25 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef _INCLUDE__BLOCK_SESSION__BLOCK_SESSION_H_
|
|
|
|
#define _INCLUDE__BLOCK_SESSION__BLOCK_SESSION_H_
|
|
|
|
|
|
|
|
#include <os/packet_stream.h>
|
|
|
|
#include <packet_stream_tx/packet_stream_tx.h>
|
|
|
|
#include <session/session.h>
|
|
|
|
|
|
|
|
namespace Block {
|
|
|
|
|
2012-05-24 21:31:02 +02:00
|
|
|
/**
|
|
|
|
* Sector type for block session
|
|
|
|
*/
|
|
|
|
typedef Genode::uint64_t sector_t;
|
|
|
|
|
2015-03-04 21:12:14 +01:00
|
|
|
class Packet_descriptor;
|
|
|
|
struct Session;
|
|
|
|
}
|
2011-12-22 16:19:25 +01:00
|
|
|
|
|
|
|
|
2015-03-04 21:12:14 +01:00
|
|
|
/**
|
2019-04-09 10:52:01 +02:00
|
|
|
* Representation of a block-operation request
|
2015-03-20 17:50:41 +01:00
|
|
|
*
|
|
|
|
* The data associated with the 'Packet_descriptor' is either
|
2015-03-04 21:12:14 +01:00
|
|
|
* the data read from or written to the block indicated by
|
|
|
|
* its number.
|
|
|
|
*/
|
2015-03-16 17:33:26 +01:00
|
|
|
class Block::Packet_descriptor : public Genode::Packet_descriptor
|
2015-03-04 21:12:14 +01:00
|
|
|
{
|
|
|
|
public:
|
2011-12-22 16:19:25 +01:00
|
|
|
|
2019-04-09 10:52:01 +02:00
|
|
|
enum Opcode { READ, WRITE, SYNC, TRIM, END };
|
2019-04-04 18:25:54 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Alignment used when allocating a packet directly via the 'tx'
|
|
|
|
* packet stream. This is not recommended because it does not
|
|
|
|
* apply the server's alignment constraints. Instead, the
|
|
|
|
* 'Block::Session_client::alloc_packet' should be used for
|
|
|
|
* allocating properly aligned block-request packets.
|
|
|
|
*/
|
2015-03-04 21:12:14 +01:00
|
|
|
enum Alignment { PACKET_ALIGNMENT = 11 };
|
2011-12-22 16:19:25 +01:00
|
|
|
|
2019-04-09 13:58:49 +02:00
|
|
|
/**
|
|
|
|
* Client-defined value for correlating acknowledgements with requests
|
|
|
|
*/
|
|
|
|
struct Tag { unsigned long value; };
|
|
|
|
|
2015-03-04 21:12:14 +01:00
|
|
|
private:
|
2011-12-22 16:19:25 +01:00
|
|
|
|
2015-03-04 21:12:14 +01:00
|
|
|
Opcode _op; /* requested operation */
|
2019-04-09 13:58:49 +02:00
|
|
|
Tag _tag; /* client-defined request identifier */
|
2015-03-04 21:12:14 +01:00
|
|
|
sector_t _block_number; /* requested block number */
|
2019-04-03 16:05:10 +02:00
|
|
|
Genode::size_t _block_count; /* number of blocks of operation */
|
|
|
|
bool _success; /* indicates success of operation */
|
2011-12-22 16:19:25 +01:00
|
|
|
|
2015-03-04 21:12:14 +01:00
|
|
|
public:
|
2011-12-22 16:19:25 +01:00
|
|
|
|
2015-03-04 21:12:14 +01:00
|
|
|
/**
|
|
|
|
* Constructor
|
|
|
|
*/
|
2015-03-16 17:33:26 +01:00
|
|
|
Packet_descriptor(Genode::off_t offset=0, Genode::size_t size = 0)
|
|
|
|
:
|
|
|
|
Genode::Packet_descriptor(offset, size),
|
2019-04-09 13:58:49 +02:00
|
|
|
_op(READ), _tag(), _block_number(0), _block_count(0), _success(false)
|
2015-03-16 17:33:26 +01:00
|
|
|
{ }
|
2011-12-22 16:19:25 +01:00
|
|
|
|
|
|
|
/**
|
2015-03-04 21:12:14 +01:00
|
|
|
* Constructor
|
2011-12-22 16:19:25 +01:00
|
|
|
*/
|
2015-03-04 21:12:14 +01:00
|
|
|
Packet_descriptor(Packet_descriptor p, Opcode op,
|
2019-04-09 13:58:49 +02:00
|
|
|
sector_t block_number, Genode::size_t block_count = 1,
|
|
|
|
Tag tag = { ~0U })
|
2015-03-16 17:33:26 +01:00
|
|
|
:
|
|
|
|
Genode::Packet_descriptor(p.offset(), p.size()),
|
2019-04-09 13:58:49 +02:00
|
|
|
_op(op), _tag(tag), _block_number(block_number),
|
2019-02-27 14:48:54 +01:00
|
|
|
_block_count(block_count), _success(false)
|
2015-03-16 17:33:26 +01:00
|
|
|
{ }
|
2011-12-22 16:19:25 +01:00
|
|
|
|
2015-03-04 21:12:14 +01:00
|
|
|
Opcode operation() const { return _op; }
|
|
|
|
sector_t block_number() const { return _block_number; }
|
|
|
|
Genode::size_t block_count() const { return _block_count; }
|
|
|
|
bool succeeded() const { return _success; }
|
2019-04-09 13:58:49 +02:00
|
|
|
Tag tag() const { return _tag; }
|
2011-12-22 16:19:25 +01:00
|
|
|
|
2019-04-03 16:05:10 +02:00
|
|
|
void succeeded(bool b) { _success = b; }
|
2015-03-04 21:12:14 +01:00
|
|
|
};
|
2011-12-22 16:19:25 +01:00
|
|
|
|
|
|
|
|
2019-04-04 18:25:54 +02:00
|
|
|
/**
|
2015-03-20 17:50:41 +01:00
|
|
|
* Block session interface
|
|
|
|
*
|
|
|
|
* A block session corresponds to a block device that can be used to read
|
|
|
|
* or store data. Payload is communicated over the packet-stream interface
|
|
|
|
* set up between 'Session_client' and 'Session_server'.
|
|
|
|
*
|
|
|
|
* Even though the methods 'tx' and 'tx_channel' are specific for the client
|
|
|
|
* side of the block session interface, they are part of the abstract 'Session'
|
|
|
|
* class to enable the client-side use of the block interface via a pointer to
|
|
|
|
* the abstract 'Session' class. This way, we can transparently co-locate the
|
|
|
|
* packet-stream server with the client in same program.
|
|
|
|
*/
|
2015-03-04 21:12:14 +01:00
|
|
|
struct Block::Session : public Genode::Session
|
|
|
|
{
|
|
|
|
enum { TX_QUEUE_SIZE = 256 };
|
2011-12-22 16:19:25 +01:00
|
|
|
|
2015-03-16 17:33:26 +01:00
|
|
|
typedef Genode::Packet_stream_policy<Block::Packet_descriptor,
|
|
|
|
TX_QUEUE_SIZE, TX_QUEUE_SIZE,
|
|
|
|
char> Tx_policy;
|
2011-12-22 16:19:25 +01:00
|
|
|
|
2015-03-04 21:12:14 +01:00
|
|
|
typedef Packet_stream_tx::Channel<Tx_policy> Tx;
|
2011-12-22 16:19:25 +01:00
|
|
|
|
2019-04-09 13:58:49 +02:00
|
|
|
typedef Packet_descriptor::Tag Tag;
|
|
|
|
|
2019-04-03 16:05:10 +02:00
|
|
|
struct Info
|
|
|
|
{
|
|
|
|
Genode::size_t block_size; /* size of one block in bytes */
|
|
|
|
sector_t block_count; /* number of blocks */
|
2019-04-04 18:25:54 +02:00
|
|
|
Genode::size_t align_log2; /* packet alignment within payload buffer */
|
2019-04-03 16:05:10 +02:00
|
|
|
bool writeable;
|
|
|
|
};
|
|
|
|
|
2017-05-24 14:41:19 +02:00
|
|
|
/**
|
|
|
|
* \noapi
|
|
|
|
*/
|
2015-03-04 21:12:14 +01:00
|
|
|
static const char *service_name() { return "Block"; }
|
2011-12-22 16:19:25 +01:00
|
|
|
|
2017-05-07 22:03:25 +02:00
|
|
|
enum { CAP_QUOTA = 5 };
|
|
|
|
|
2015-03-04 21:12:14 +01:00
|
|
|
virtual ~Session() { }
|
2011-12-22 16:19:25 +01:00
|
|
|
|
2015-03-04 21:12:14 +01:00
|
|
|
/**
|
|
|
|
* Request information about the metrics of the block device
|
|
|
|
*/
|
2019-04-03 16:05:10 +02:00
|
|
|
virtual Info info() const = 0;
|
2015-03-04 21:12:14 +01:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Request packet-transmission channel
|
|
|
|
*/
|
|
|
|
virtual Tx *tx_channel() { return 0; }
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Request client-side packet-stream interface of tx channel
|
|
|
|
*/
|
|
|
|
virtual Tx::Source *tx() { return 0; }
|
|
|
|
|
2018-12-12 21:22:19 +01:00
|
|
|
/**
|
|
|
|
* Return capability for packet-transmission channel
|
|
|
|
*/
|
|
|
|
virtual Genode::Capability<Tx> tx_cap() = 0;
|
|
|
|
|
2019-04-09 10:52:01 +02:00
|
|
|
/**
|
|
|
|
* Return packet descriptor for syncing the entire block session
|
|
|
|
*/
|
|
|
|
static Packet_descriptor sync_all_packet_descriptor(Info const &info, Tag tag)
|
|
|
|
{
|
|
|
|
return Packet_descriptor(Packet_descriptor(0UL, 0UL),
|
|
|
|
Packet_descriptor::SYNC,
|
|
|
|
0UL, info.block_count, tag);
|
|
|
|
}
|
|
|
|
|
2015-03-04 21:12:14 +01:00
|
|
|
|
|
|
|
/*******************
|
|
|
|
** RPC interface **
|
|
|
|
*******************/
|
|
|
|
|
2019-04-03 16:05:10 +02:00
|
|
|
GENODE_RPC(Rpc_info, Info, info);
|
2018-12-12 21:22:19 +01:00
|
|
|
GENODE_RPC(Rpc_tx_cap, Genode::Capability<Tx>, tx_cap);
|
2019-04-09 10:52:01 +02:00
|
|
|
GENODE_RPC_INTERFACE(Rpc_info, Rpc_tx_cap);
|
2015-03-04 21:12:14 +01:00
|
|
|
};
|
2011-12-22 16:19:25 +01:00
|
|
|
|
|
|
|
#endif /* _INCLUDE__BLOCK_SESSION__BLOCK_SESSION_H_ */
|