mirror of
https://github.com/zeromq/libzmq.git
synced 2024-12-13 02:42:58 +01:00
SWAP functionality removed
On-disk storage should be implemented in devices rather than in 0MQ core. 0MQ is a networking library and there's no point in storing network buffers on disk. Signed-off-by: Martin Sustrik <sustrik@250bpm.com>
This commit is contained in:
parent
fb27a000d9
commit
8463b4d55e
@ -79,22 +79,6 @@ Default value:: 0
|
|||||||
Applicable socket types:: all
|
Applicable socket types:: all
|
||||||
|
|
||||||
|
|
||||||
ZMQ_SWAP: Retrieve disk offload size
|
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
||||||
The 'ZMQ_SWAP' option shall retrieve the disk offload (swap) size for the
|
|
||||||
specified 'socket'. A socket which has 'ZMQ_SWAP' set to a non-zero value may
|
|
||||||
exceed it's high water mark; in this case outstanding messages shall be
|
|
||||||
offloaded to storage on disk rather than held in memory.
|
|
||||||
|
|
||||||
The value of 'ZMQ_SWAP' defines the maximum size of the swap space in bytes.
|
|
||||||
|
|
||||||
[horizontal]
|
|
||||||
Option value type:: int64_t
|
|
||||||
Option value unit:: bytes
|
|
||||||
Default value:: 0
|
|
||||||
Applicable socket types:: all
|
|
||||||
|
|
||||||
|
|
||||||
ZMQ_AFFINITY: Retrieve I/O thread affinity
|
ZMQ_AFFINITY: Retrieve I/O thread affinity
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
The 'ZMQ_AFFINITY' option shall retrieve the I/O thread affinity for newly
|
The 'ZMQ_AFFINITY' option shall retrieve the I/O thread affinity for newly
|
||||||
|
@ -47,22 +47,6 @@ Default value:: 0
|
|||||||
Applicable socket types:: all
|
Applicable socket types:: all
|
||||||
|
|
||||||
|
|
||||||
ZMQ_SWAP: Set disk offload size
|
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
||||||
The 'ZMQ_SWAP' option shall set the disk offload (swap) size for the specified
|
|
||||||
'socket'. A socket which has 'ZMQ_SWAP' set to a non-zero value may exceed it's
|
|
||||||
high water mark; in this case outstanding messages shall be offloaded to
|
|
||||||
storage on disk rather than held in memory.
|
|
||||||
|
|
||||||
The value of 'ZMQ_SWAP' defines the maximum size of the swap space in bytes.
|
|
||||||
|
|
||||||
[horizontal]
|
|
||||||
Option value type:: int64_t
|
|
||||||
Option value unit:: bytes
|
|
||||||
Default value:: 0
|
|
||||||
Applicable socket types:: all
|
|
||||||
|
|
||||||
|
|
||||||
ZMQ_AFFINITY: Set I/O thread affinity
|
ZMQ_AFFINITY: Set I/O thread affinity
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
The 'ZMQ_AFFINITY' option shall set the I/O thread affinity for newly created
|
The 'ZMQ_AFFINITY' option shall set the I/O thread affinity for newly created
|
||||||
|
@ -184,7 +184,6 @@ ZMQ_EXPORT int zmq_term (void *context);
|
|||||||
|
|
||||||
/* Socket options. */
|
/* Socket options. */
|
||||||
#define ZMQ_HWM 1
|
#define ZMQ_HWM 1
|
||||||
#define ZMQ_SWAP 3
|
|
||||||
#define ZMQ_AFFINITY 4
|
#define ZMQ_AFFINITY 4
|
||||||
#define ZMQ_IDENTITY 5
|
#define ZMQ_IDENTITY 5
|
||||||
#define ZMQ_SUBSCRIBE 6
|
#define ZMQ_SUBSCRIBE 6
|
||||||
|
@ -115,7 +115,6 @@ libzmq_la_SOURCES = \
|
|||||||
socket_base.hpp \
|
socket_base.hpp \
|
||||||
stdint.hpp \
|
stdint.hpp \
|
||||||
sub.hpp \
|
sub.hpp \
|
||||||
swap.hpp \
|
|
||||||
tcp_connecter.hpp \
|
tcp_connecter.hpp \
|
||||||
tcp_listener.hpp \
|
tcp_listener.hpp \
|
||||||
tcp_socket.hpp \
|
tcp_socket.hpp \
|
||||||
@ -173,7 +172,6 @@ libzmq_la_SOURCES = \
|
|||||||
session.cpp \
|
session.cpp \
|
||||||
socket_base.cpp \
|
socket_base.cpp \
|
||||||
sub.cpp \
|
sub.cpp \
|
||||||
swap.cpp \
|
|
||||||
tcp_connecter.cpp \
|
tcp_connecter.cpp \
|
||||||
tcp_listener.cpp \
|
tcp_listener.cpp \
|
||||||
tcp_socket.cpp \
|
tcp_socket.cpp \
|
||||||
|
@ -59,10 +59,6 @@ namespace zmq
|
|||||||
// Maximal delta between high and low watermark.
|
// Maximal delta between high and low watermark.
|
||||||
max_wm_delta = 1024,
|
max_wm_delta = 1024,
|
||||||
|
|
||||||
// Swap inteligently batches data for writing to disk. The size of
|
|
||||||
// the batch in bytes is specified by this option.
|
|
||||||
swap_block_size = 8192,
|
|
||||||
|
|
||||||
// Maximum number of events the I/O thread can process in one go.
|
// Maximum number of events the I/O thread can process in one go.
|
||||||
max_io_events = 256,
|
max_io_events = 256,
|
||||||
|
|
||||||
|
@ -27,7 +27,6 @@
|
|||||||
|
|
||||||
zmq::options_t::options_t () :
|
zmq::options_t::options_t () :
|
||||||
hwm (0),
|
hwm (0),
|
||||||
swap (0),
|
|
||||||
affinity (0),
|
affinity (0),
|
||||||
rate (100),
|
rate (100),
|
||||||
recovery_ivl (10),
|
recovery_ivl (10),
|
||||||
@ -59,14 +58,6 @@ int zmq::options_t::setsockopt (int option_, const void *optval_,
|
|||||||
hwm = *((uint64_t*) optval_);
|
hwm = *((uint64_t*) optval_);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
case ZMQ_SWAP:
|
|
||||||
if (optvallen_ != sizeof (int64_t) || *((int64_t*) optval_) < 0) {
|
|
||||||
errno = EINVAL;
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
swap = *((int64_t*) optval_);
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
case ZMQ_AFFINITY:
|
case ZMQ_AFFINITY:
|
||||||
if (optvallen_ != sizeof (uint64_t)) {
|
if (optvallen_ != sizeof (uint64_t)) {
|
||||||
errno = EINVAL;
|
errno = EINVAL;
|
||||||
@ -195,15 +186,6 @@ int zmq::options_t::getsockopt (int option_, void *optval_, size_t *optvallen_)
|
|||||||
*optvallen_ = sizeof (uint64_t);
|
*optvallen_ = sizeof (uint64_t);
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
case ZMQ_SWAP:
|
|
||||||
if (*optvallen_ < sizeof (int64_t)) {
|
|
||||||
errno = EINVAL;
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
*((int64_t*) optval_) = swap;
|
|
||||||
*optvallen_ = sizeof (int64_t);
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
case ZMQ_AFFINITY:
|
case ZMQ_AFFINITY:
|
||||||
if (*optvallen_ < sizeof (uint64_t)) {
|
if (*optvallen_ < sizeof (uint64_t)) {
|
||||||
errno = EINVAL;
|
errno = EINVAL;
|
||||||
|
@ -36,7 +36,6 @@ namespace zmq
|
|||||||
int getsockopt (int option_, void *optval_, size_t *optvallen_);
|
int getsockopt (int option_, void *optval_, size_t *optvallen_);
|
||||||
|
|
||||||
uint64_t hwm;
|
uint64_t hwm;
|
||||||
int64_t swap;
|
|
||||||
uint64_t affinity;
|
uint64_t affinity;
|
||||||
blob_t identity;
|
blob_t identity;
|
||||||
|
|
||||||
|
87
src/pipe.cpp
87
src/pipe.cpp
@ -163,7 +163,7 @@ void zmq::reader_t::process_pipe_term_ack ()
|
|||||||
}
|
}
|
||||||
|
|
||||||
zmq::writer_t::writer_t (object_t *parent_, pipe_t *pipe_, reader_t *reader_,
|
zmq::writer_t::writer_t (object_t *parent_, pipe_t *pipe_, reader_t *reader_,
|
||||||
uint64_t hwm_, int64_t swap_size_) :
|
uint64_t hwm_) :
|
||||||
object_t (parent_),
|
object_t (parent_),
|
||||||
active (true),
|
active (true),
|
||||||
pipe (pipe_),
|
pipe (pipe_),
|
||||||
@ -171,28 +171,15 @@ zmq::writer_t::writer_t (object_t *parent_, pipe_t *pipe_, reader_t *reader_,
|
|||||||
hwm (hwm_),
|
hwm (hwm_),
|
||||||
msgs_read (0),
|
msgs_read (0),
|
||||||
msgs_written (0),
|
msgs_written (0),
|
||||||
swap (NULL),
|
|
||||||
sink (NULL),
|
sink (NULL),
|
||||||
swapping (false),
|
|
||||||
pending_delimiter (false),
|
|
||||||
terminating (false)
|
terminating (false)
|
||||||
{
|
{
|
||||||
// Inform reader about the writer.
|
// Inform reader about the writer.
|
||||||
reader->set_writer (this);
|
reader->set_writer (this);
|
||||||
|
|
||||||
// Open the swap file, if required.
|
|
||||||
if (swap_size_ > 0) {
|
|
||||||
swap = new (std::nothrow) swap_t (swap_size_);
|
|
||||||
alloc_assert (swap);
|
|
||||||
int rc = swap->init ();
|
|
||||||
zmq_assert (rc == 0);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
zmq::writer_t::~writer_t ()
|
zmq::writer_t::~writer_t ()
|
||||||
{
|
{
|
||||||
if (swap)
|
|
||||||
delete swap;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void zmq::writer_t::set_event_sink (i_writer_events *sink_)
|
void zmq::writer_t::set_event_sink (i_writer_events *sink_)
|
||||||
@ -208,22 +195,10 @@ bool zmq::writer_t::check_write (zmq_msg_t *msg_)
|
|||||||
if (unlikely (!active))
|
if (unlikely (!active))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (unlikely (swapping)) {
|
|
||||||
if (unlikely (!swap->fits (msg_))) {
|
|
||||||
active = false;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
if (unlikely (pipe_full ())) {
|
if (unlikely (pipe_full ())) {
|
||||||
if (swap)
|
|
||||||
swapping = true;
|
|
||||||
else {
|
|
||||||
active = false;
|
active = false;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -233,14 +208,6 @@ bool zmq::writer_t::write (zmq_msg_t *msg_)
|
|||||||
if (unlikely (!check_write (msg_)))
|
if (unlikely (!check_write (msg_)))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (unlikely (swapping)) {
|
|
||||||
bool stored = swap->store (msg_);
|
|
||||||
zmq_assert (stored);
|
|
||||||
if (!(msg_->flags & ZMQ_MSG_MORE))
|
|
||||||
swap->commit ();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
pipe->write (*msg_, msg_->flags & ZMQ_MSG_MORE);
|
pipe->write (*msg_, msg_->flags & ZMQ_MSG_MORE);
|
||||||
if (!(msg_->flags & ZMQ_MSG_MORE))
|
if (!(msg_->flags & ZMQ_MSG_MORE))
|
||||||
msgs_written++;
|
msgs_written++;
|
||||||
@ -250,12 +217,6 @@ bool zmq::writer_t::write (zmq_msg_t *msg_)
|
|||||||
|
|
||||||
void zmq::writer_t::rollback ()
|
void zmq::writer_t::rollback ()
|
||||||
{
|
{
|
||||||
// Remove incomplete message from the swap.
|
|
||||||
if (unlikely (swapping)) {
|
|
||||||
swap->rollback ();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove incomplete message from the pipe.
|
// Remove incomplete message from the pipe.
|
||||||
zmq_msg_t msg;
|
zmq_msg_t msg;
|
||||||
while (pipe->unwrite (&msg)) {
|
while (pipe->unwrite (&msg)) {
|
||||||
@ -266,8 +227,7 @@ void zmq::writer_t::rollback ()
|
|||||||
|
|
||||||
void zmq::writer_t::flush ()
|
void zmq::writer_t::flush ()
|
||||||
{
|
{
|
||||||
// In the swapping mode, flushing is automatically handled by swap object.
|
if (!pipe->flush ())
|
||||||
if (!swapping && !pipe->flush ())
|
|
||||||
send_activate_reader (reader);
|
send_activate_reader (reader);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -284,11 +244,6 @@ void zmq::writer_t::terminate ()
|
|||||||
// Rollback any unfinished messages.
|
// Rollback any unfinished messages.
|
||||||
rollback ();
|
rollback ();
|
||||||
|
|
||||||
if (swapping) {
|
|
||||||
pending_delimiter = true;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Push delimiter into the pipe. Trick the compiler to belive that
|
// Push delimiter into the pipe. Trick the compiler to belive that
|
||||||
// the tag is a valid pointer. Note that watermarks are not checked
|
// the tag is a valid pointer. Note that watermarks are not checked
|
||||||
// thus the delimiter can be written even though the pipe is full.
|
// thus the delimiter can be written even though the pipe is full.
|
||||||
@ -305,40 +260,6 @@ void zmq::writer_t::process_activate_writer (uint64_t msgs_read_)
|
|||||||
// Store the reader's message sequence number.
|
// Store the reader's message sequence number.
|
||||||
msgs_read = msgs_read_;
|
msgs_read = msgs_read_;
|
||||||
|
|
||||||
// If we are in the swapping mode, we have some messages in the swap.
|
|
||||||
// Given that pipe is now ready for writing we can move part of the
|
|
||||||
// swap into the pipe.
|
|
||||||
if (swapping) {
|
|
||||||
zmq_msg_t msg;
|
|
||||||
while (!pipe_full () && !swap->empty ()) {
|
|
||||||
swap->fetch(&msg);
|
|
||||||
pipe->write (msg, msg.flags & ZMQ_MSG_MORE);
|
|
||||||
if (!(msg.flags & ZMQ_MSG_MORE))
|
|
||||||
msgs_written++;
|
|
||||||
}
|
|
||||||
if (!pipe->flush ())
|
|
||||||
send_activate_reader (reader);
|
|
||||||
|
|
||||||
// There are no more messages in the swap. We can switch into
|
|
||||||
// standard in-memory mode.
|
|
||||||
if (swap->empty ()) {
|
|
||||||
swapping = false;
|
|
||||||
|
|
||||||
// Push delimiter into the pipe. Trick the compiler to belive that
|
|
||||||
// the tag is a valid pointer. Note that watermarks are not checked
|
|
||||||
// thus the delimiter can be written even though the pipe is full.
|
|
||||||
if (pending_delimiter) {
|
|
||||||
zmq_msg_t msg;
|
|
||||||
const unsigned char *offset = 0;
|
|
||||||
msg.content = (void*) (offset + ZMQ_DELIMITER);
|
|
||||||
msg.flags = 0;
|
|
||||||
pipe->write (msg, false);
|
|
||||||
flush ();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the writer was non-active before, let's make it active
|
// If the writer was non-active before, let's make it active
|
||||||
// (available for writing messages to).
|
// (available for writing messages to).
|
||||||
if (!active && !terminating) {
|
if (!active && !terminating) {
|
||||||
@ -371,7 +292,7 @@ bool zmq::writer_t::pipe_full ()
|
|||||||
}
|
}
|
||||||
|
|
||||||
void zmq::create_pipe (object_t *reader_parent_, object_t *writer_parent_,
|
void zmq::create_pipe (object_t *reader_parent_, object_t *writer_parent_,
|
||||||
uint64_t hwm_, int64_t swap_size_, reader_t **reader_, writer_t **writer_)
|
uint64_t hwm_, reader_t **reader_, writer_t **writer_)
|
||||||
{
|
{
|
||||||
// First compute the low water mark. Following point should be taken
|
// First compute the low water mark. Following point should be taken
|
||||||
// into consideration:
|
// into consideration:
|
||||||
@ -404,6 +325,6 @@ void zmq::create_pipe (object_t *reader_parent_, object_t *writer_parent_,
|
|||||||
*reader_ = new (std::nothrow) reader_t (reader_parent_, pipe, lwm);
|
*reader_ = new (std::nothrow) reader_t (reader_parent_, pipe, lwm);
|
||||||
alloc_assert (*reader_);
|
alloc_assert (*reader_);
|
||||||
*writer_ = new (std::nothrow) writer_t (writer_parent_, pipe, *reader_,
|
*writer_ = new (std::nothrow) writer_t (writer_parent_, pipe, *reader_,
|
||||||
hwm_, swap_size_);
|
hwm_);
|
||||||
alloc_assert (*writer_);
|
alloc_assert (*writer_);
|
||||||
}
|
}
|
||||||
|
30
src/pipe.hpp
30
src/pipe.hpp
@ -26,7 +26,6 @@
|
|||||||
#include "stdint.hpp"
|
#include "stdint.hpp"
|
||||||
#include "array.hpp"
|
#include "array.hpp"
|
||||||
#include "ypipe.hpp"
|
#include "ypipe.hpp"
|
||||||
#include "swap.hpp"
|
|
||||||
#include "config.hpp"
|
#include "config.hpp"
|
||||||
#include "object.hpp"
|
#include "object.hpp"
|
||||||
|
|
||||||
@ -35,7 +34,7 @@ namespace zmq
|
|||||||
|
|
||||||
// Creates a pipe. Returns pointer to reader and writer objects.
|
// Creates a pipe. Returns pointer to reader and writer objects.
|
||||||
void create_pipe (object_t *reader_parent_, object_t *writer_parent_,
|
void create_pipe (object_t *reader_parent_, object_t *writer_parent_,
|
||||||
uint64_t hwm_, int64_t swap_size_, class reader_t **reader_,
|
uint64_t hwm_, class reader_t **reader_,
|
||||||
class writer_t **writer_);
|
class writer_t **writer_);
|
||||||
|
|
||||||
// The shutdown mechanism for pipe works as follows: Either endpoint
|
// The shutdown mechanism for pipe works as follows: Either endpoint
|
||||||
@ -59,7 +58,7 @@ namespace zmq
|
|||||||
class reader_t : public object_t, public array_item_t
|
class reader_t : public object_t, public array_item_t
|
||||||
{
|
{
|
||||||
friend void create_pipe (object_t*, object_t*, uint64_t,
|
friend void create_pipe (object_t*, object_t*, uint64_t,
|
||||||
int64_t, reader_t**, writer_t**);
|
reader_t**, writer_t**);
|
||||||
friend class writer_t;
|
friend class writer_t;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
@ -128,7 +127,7 @@ namespace zmq
|
|||||||
class writer_t : public object_t, public array_item_t
|
class writer_t : public object_t, public array_item_t
|
||||||
{
|
{
|
||||||
friend void create_pipe (object_t*, object_t*, uint64_t,
|
friend void create_pipe (object_t*, object_t*, uint64_t,
|
||||||
int64_t, reader_t**, writer_t**);
|
reader_t**, writer_t**);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
@ -136,8 +135,8 @@ namespace zmq
|
|||||||
void set_event_sink (i_writer_events *endpoint_);
|
void set_event_sink (i_writer_events *endpoint_);
|
||||||
|
|
||||||
// Checks whether messages can be written to the pipe.
|
// Checks whether messages can be written to the pipe.
|
||||||
// If writing the message would cause high watermark and (optionally)
|
// If writing the message would cause high watermark
|
||||||
// if the swap is full, the function returns false.
|
// the function returns false.
|
||||||
bool check_write (zmq_msg_t *msg_);
|
bool check_write (zmq_msg_t *msg_);
|
||||||
|
|
||||||
// Writes a message to the underlying pipe. Returns false if the
|
// Writes a message to the underlying pipe. Returns false if the
|
||||||
@ -156,19 +155,17 @@ namespace zmq
|
|||||||
private:
|
private:
|
||||||
|
|
||||||
writer_t (class object_t *parent_, pipe_t *pipe_, reader_t *reader_,
|
writer_t (class object_t *parent_, pipe_t *pipe_, reader_t *reader_,
|
||||||
uint64_t hwm_, int64_t swap_size_);
|
uint64_t hwm_);
|
||||||
~writer_t ();
|
~writer_t ();
|
||||||
|
|
||||||
// Command handlers.
|
// Command handlers.
|
||||||
void process_activate_writer (uint64_t msgs_read_);
|
void process_activate_writer (uint64_t msgs_read_);
|
||||||
void process_pipe_term ();
|
void process_pipe_term ();
|
||||||
|
|
||||||
// Tests whether underlying pipe is already full. The swap is not
|
// Tests whether underlying pipe is already full.
|
||||||
// taken into account.
|
|
||||||
bool pipe_full ();
|
bool pipe_full ();
|
||||||
|
|
||||||
// True, if this object can be written to. Undelying ypipe may be full
|
// True, if this object can be written to.
|
||||||
// but as long as there's swap space available, this flag is true.
|
|
||||||
bool active;
|
bool active;
|
||||||
|
|
||||||
// The underlying pipe.
|
// The underlying pipe.
|
||||||
@ -187,20 +184,9 @@ namespace zmq
|
|||||||
// Number of messages we have written so far.
|
// Number of messages we have written so far.
|
||||||
uint64_t msgs_written;
|
uint64_t msgs_written;
|
||||||
|
|
||||||
// Pointer to the message swap. If NULL, messages are always
|
|
||||||
// kept in main memory.
|
|
||||||
swap_t *swap;
|
|
||||||
|
|
||||||
// Sink for the events (either the socket or the session).
|
// Sink for the events (either the socket or the session).
|
||||||
i_writer_events *sink;
|
i_writer_events *sink;
|
||||||
|
|
||||||
// If true, swap is active. New messages are to be written to the swap.
|
|
||||||
bool swapping;
|
|
||||||
|
|
||||||
// If true, there's a delimiter to be written to the pipe after the
|
|
||||||
// swap is empied.
|
|
||||||
bool pending_delimiter;
|
|
||||||
|
|
||||||
// True is 'terminate' method was called of 'pipe_term' command
|
// True is 'terminate' method was called of 'pipe_term' command
|
||||||
// arrived from the reader.
|
// arrived from the reader.
|
||||||
bool terminating;
|
bool terminating;
|
||||||
|
@ -247,13 +247,11 @@ void zmq::session_t::process_attach (i_engine *engine_,
|
|||||||
|
|
||||||
// Create the pipes, as required.
|
// Create the pipes, as required.
|
||||||
if (options.requires_in) {
|
if (options.requires_in) {
|
||||||
create_pipe (socket, this, options.hwm, options.swap,
|
create_pipe (socket, this, options.hwm, &socket_reader, &out_pipe);
|
||||||
&socket_reader, &out_pipe);
|
|
||||||
out_pipe->set_event_sink (this);
|
out_pipe->set_event_sink (this);
|
||||||
}
|
}
|
||||||
if (options.requires_out) {
|
if (options.requires_out) {
|
||||||
create_pipe (this, socket, options.hwm, options.swap, &in_pipe,
|
create_pipe (this, socket, options.hwm, &in_pipe, &socket_writer);
|
||||||
&socket_writer);
|
|
||||||
in_pipe->set_event_sink (this);
|
in_pipe->set_event_sink (this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -376,28 +376,22 @@ int zmq::socket_base_t::connect (const char *addr_)
|
|||||||
writer_t *outpipe_writer = NULL;
|
writer_t *outpipe_writer = NULL;
|
||||||
|
|
||||||
// The total HWM for an inproc connection should be the sum of
|
// The total HWM for an inproc connection should be the sum of
|
||||||
// the binder's HWM and the connector's HWM. (Similarly for the
|
// the binder's HWM and the connector's HWM.
|
||||||
// SWAP.)
|
|
||||||
int64_t hwm;
|
int64_t hwm;
|
||||||
if (options.hwm == 0 || peer.options.hwm == 0)
|
if (options.hwm == 0 || peer.options.hwm == 0)
|
||||||
hwm = 0;
|
hwm = 0;
|
||||||
else
|
else
|
||||||
hwm = options.hwm + peer.options.hwm;
|
hwm = options.hwm + peer.options.hwm;
|
||||||
int64_t swap;
|
|
||||||
if (options.swap == 0 && peer.options.swap == 0)
|
|
||||||
swap = 0;
|
|
||||||
else
|
|
||||||
swap = options.swap + peer.options.swap;
|
|
||||||
|
|
||||||
// Create inbound pipe, if required.
|
// Create inbound pipe, if required.
|
||||||
if (options.requires_in)
|
if (options.requires_in)
|
||||||
create_pipe (this, peer.socket, hwm, swap,
|
create_pipe (this, peer.socket, hwm, &inpipe_reader,
|
||||||
&inpipe_reader, &inpipe_writer);
|
&inpipe_writer);
|
||||||
|
|
||||||
// Create outbound pipe, if required.
|
// Create outbound pipe, if required.
|
||||||
if (options.requires_out)
|
if (options.requires_out)
|
||||||
create_pipe (peer.socket, this, hwm, swap,
|
create_pipe (peer.socket, this, hwm, &outpipe_reader,
|
||||||
&outpipe_reader, &outpipe_writer);
|
&outpipe_writer);
|
||||||
|
|
||||||
// Attach the pipes to this socket object.
|
// Attach the pipes to this socket object.
|
||||||
attach_pipes (inpipe_reader, outpipe_writer, peer.options.identity);
|
attach_pipes (inpipe_reader, outpipe_writer, peer.options.identity);
|
||||||
@ -435,12 +429,12 @@ int zmq::socket_base_t::connect (const char *addr_)
|
|||||||
|
|
||||||
// Create inbound pipe, if required.
|
// Create inbound pipe, if required.
|
||||||
if (options.requires_in)
|
if (options.requires_in)
|
||||||
create_pipe (this, session, options.hwm, options.swap,
|
create_pipe (this, session, options.hwm,
|
||||||
&inpipe_reader, &inpipe_writer);
|
&inpipe_reader, &inpipe_writer);
|
||||||
|
|
||||||
// Create outbound pipe, if required.
|
// Create outbound pipe, if required.
|
||||||
if (options.requires_out)
|
if (options.requires_out)
|
||||||
create_pipe (session, this, options.hwm, options.swap,
|
create_pipe (session, this, options.hwm,
|
||||||
&outpipe_reader, &outpipe_writer);
|
&outpipe_reader, &outpipe_writer);
|
||||||
|
|
||||||
// Attach the pipes to the socket object.
|
// Attach the pipes to the socket object.
|
||||||
|
325
src/swap.cpp
325
src/swap.cpp
@ -1,325 +0,0 @@
|
|||||||
/*
|
|
||||||
Copyright (c) 2007-2011 iMatix Corporation
|
|
||||||
Copyright (c) 2007-2011 Other contributors as noted in the AUTHORS file
|
|
||||||
|
|
||||||
This file is part of 0MQ.
|
|
||||||
|
|
||||||
0MQ is free software; you can redistribute it and/or modify it under
|
|
||||||
the terms of the GNU Lesser General Public License as published by
|
|
||||||
the Free Software Foundation; either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
0MQ is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU Lesser General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU Lesser General Public License
|
|
||||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "platform.hpp"
|
|
||||||
|
|
||||||
#ifdef ZMQ_HAVE_WINDOWS
|
|
||||||
#include "windows.hpp"
|
|
||||||
#include <io.h>
|
|
||||||
#else
|
|
||||||
#include <unistd.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#include "../include/zmq.h"
|
|
||||||
|
|
||||||
#include <sys/types.h>
|
|
||||||
#include <sys/stat.h>
|
|
||||||
#include <fcntl.h>
|
|
||||||
#include <string.h>
|
|
||||||
#include <sstream>
|
|
||||||
#include <algorithm>
|
|
||||||
|
|
||||||
#include "swap.hpp"
|
|
||||||
#include "config.hpp"
|
|
||||||
#include "atomic_counter.hpp"
|
|
||||||
#include "err.hpp"
|
|
||||||
|
|
||||||
zmq::swap_t::swap_t (int64_t filesize_) :
|
|
||||||
fd (-1),
|
|
||||||
filesize (filesize_),
|
|
||||||
file_pos (0),
|
|
||||||
write_pos (0),
|
|
||||||
read_pos (0),
|
|
||||||
block_size (swap_block_size),
|
|
||||||
write_buf_start_addr (0)
|
|
||||||
{
|
|
||||||
zmq_assert (filesize > 0);
|
|
||||||
zmq_assert (block_size > 0);
|
|
||||||
|
|
||||||
buf1 = new (std::nothrow) char [block_size];
|
|
||||||
alloc_assert (buf1);
|
|
||||||
|
|
||||||
buf2 = new (std::nothrow) char [block_size];
|
|
||||||
alloc_assert (buf2);
|
|
||||||
|
|
||||||
read_buf = write_buf = buf1;
|
|
||||||
}
|
|
||||||
|
|
||||||
zmq::swap_t::~swap_t ()
|
|
||||||
{
|
|
||||||
delete [] buf1;
|
|
||||||
delete [] buf2;
|
|
||||||
|
|
||||||
if (fd == -1)
|
|
||||||
return;
|
|
||||||
|
|
||||||
#ifdef ZMQ_HAVE_WINDOWS
|
|
||||||
int rc = _close (fd);
|
|
||||||
#else
|
|
||||||
int rc = close (fd);
|
|
||||||
#endif
|
|
||||||
errno_assert (rc == 0);
|
|
||||||
|
|
||||||
#ifdef ZMQ_HAVE_WINDOWS
|
|
||||||
rc = _unlink (filename.c_str ());
|
|
||||||
#else
|
|
||||||
rc = unlink (filename.c_str ());
|
|
||||||
#endif
|
|
||||||
errno_assert (rc == 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
int zmq::swap_t::init ()
|
|
||||||
{
|
|
||||||
static zmq::atomic_counter_t seqnum (0);
|
|
||||||
|
|
||||||
// Get process ID.
|
|
||||||
#ifdef ZMQ_HAVE_WINDOWS
|
|
||||||
int pid = GetCurrentThreadId ();
|
|
||||||
#else
|
|
||||||
pid_t pid = getpid ();
|
|
||||||
#endif
|
|
||||||
|
|
||||||
std::ostringstream outs;
|
|
||||||
outs << "zmq_" << pid << '_' << seqnum.get () << ".swap";
|
|
||||||
filename = outs.str ();
|
|
||||||
|
|
||||||
seqnum.add (1);
|
|
||||||
|
|
||||||
// Open the backing file.
|
|
||||||
#ifdef ZMQ_HAVE_WINDOWS
|
|
||||||
fd = _open (filename.c_str (), _O_RDWR | _O_CREAT, 0600);
|
|
||||||
#else
|
|
||||||
fd = open (filename.c_str (), O_RDWR | O_CREAT, 0600);
|
|
||||||
#endif
|
|
||||||
if (fd == -1)
|
|
||||||
return -1;
|
|
||||||
|
|
||||||
#ifdef ZMQ_HAVE_LINUX
|
|
||||||
// Enable more aggresive read-ahead optimization.
|
|
||||||
posix_fadvise (fd, 0, filesize, POSIX_FADV_SEQUENTIAL);
|
|
||||||
#endif
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool zmq::swap_t::store (zmq_msg_t *msg_)
|
|
||||||
{
|
|
||||||
size_t msg_size = zmq_msg_size (msg_);
|
|
||||||
|
|
||||||
// Check buffer space availability.
|
|
||||||
// NOTE: We always keep one byte open.
|
|
||||||
if (buffer_space () <= (int64_t) (sizeof msg_size + 1 + msg_size))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
// Don't store the ZMQ_MSG_SHARED flag.
|
|
||||||
uint8_t msg_flags = msg_->flags & ~ZMQ_MSG_SHARED;
|
|
||||||
|
|
||||||
// Write message length, flags, and message body.
|
|
||||||
copy_to_file (&msg_size, sizeof msg_size);
|
|
||||||
copy_to_file (&msg_flags, sizeof msg_flags);
|
|
||||||
copy_to_file (zmq_msg_data (msg_), msg_size);
|
|
||||||
|
|
||||||
zmq_msg_close (msg_);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void zmq::swap_t::fetch (zmq_msg_t *msg_)
|
|
||||||
{
|
|
||||||
// There must be at least one message available.
|
|
||||||
zmq_assert (read_pos != write_pos);
|
|
||||||
|
|
||||||
// Retrieve the message size.
|
|
||||||
size_t msg_size;
|
|
||||||
copy_from_file (&msg_size, sizeof msg_size);
|
|
||||||
|
|
||||||
// Initialize the message.
|
|
||||||
zmq_msg_init_size (msg_, msg_size);
|
|
||||||
|
|
||||||
// Retrieve the message flags.
|
|
||||||
copy_from_file (&msg_->flags, sizeof msg_->flags);
|
|
||||||
|
|
||||||
// Retrieve the message payload.
|
|
||||||
copy_from_file (zmq_msg_data (msg_), msg_size);
|
|
||||||
}
|
|
||||||
|
|
||||||
void zmq::swap_t::commit ()
|
|
||||||
{
|
|
||||||
commit_pos = write_pos;
|
|
||||||
}
|
|
||||||
|
|
||||||
void zmq::swap_t::rollback ()
|
|
||||||
{
|
|
||||||
if (commit_pos == write_pos || read_pos == write_pos)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (write_pos > read_pos)
|
|
||||||
zmq_assert (read_pos <= commit_pos && commit_pos <= write_pos);
|
|
||||||
else
|
|
||||||
zmq_assert (read_pos <= commit_pos || commit_pos <= write_pos);
|
|
||||||
|
|
||||||
if (commit_pos / block_size == read_pos / block_size) {
|
|
||||||
write_buf_start_addr = commit_pos % block_size;
|
|
||||||
write_buf = read_buf;
|
|
||||||
}
|
|
||||||
else if (commit_pos / block_size != write_pos / block_size) {
|
|
||||||
write_buf_start_addr = commit_pos % block_size;
|
|
||||||
fill_buf (write_buf, write_buf_start_addr);
|
|
||||||
}
|
|
||||||
write_pos = commit_pos;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool zmq::swap_t::empty ()
|
|
||||||
{
|
|
||||||
return read_pos == write_pos;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
bool zmq::swap_t::full ()
|
|
||||||
{
|
|
||||||
// Check that at least the message size can be written to the swap.
|
|
||||||
return buffer_space () < (int64_t) (sizeof (size_t) + 1);
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
bool zmq::swap_t::fits (zmq_msg_t *msg_)
|
|
||||||
{
|
|
||||||
// Check whether whole binary representation of the message
|
|
||||||
// fits into the swap.
|
|
||||||
size_t msg_size = zmq_msg_size (msg_);
|
|
||||||
if (buffer_space () <= (int64_t) (sizeof msg_size + 1 + msg_size))
|
|
||||||
return false;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void zmq::swap_t::copy_from_file (void *buffer_, size_t count_)
|
|
||||||
{
|
|
||||||
char *dest_ptr = (char *) buffer_;
|
|
||||||
size_t chunk_size, remainder = count_;
|
|
||||||
|
|
||||||
while (remainder > 0) {
|
|
||||||
chunk_size = std::min (remainder,
|
|
||||||
std::min ((size_t) (filesize - read_pos),
|
|
||||||
(size_t) (block_size - read_pos % block_size)));
|
|
||||||
|
|
||||||
memcpy (dest_ptr, &read_buf [read_pos % block_size], chunk_size);
|
|
||||||
dest_ptr += chunk_size;
|
|
||||||
|
|
||||||
read_pos = (read_pos + chunk_size) % filesize;
|
|
||||||
if (read_pos % block_size == 0) {
|
|
||||||
if (read_pos / block_size == write_pos / block_size)
|
|
||||||
read_buf = write_buf;
|
|
||||||
else
|
|
||||||
fill_buf (read_buf, read_pos);
|
|
||||||
}
|
|
||||||
remainder -= chunk_size;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void zmq::swap_t::copy_to_file (const void *buffer_, size_t count_)
|
|
||||||
{
|
|
||||||
char *source_ptr = (char *) buffer_;
|
|
||||||
size_t chunk_size, remainder = count_;
|
|
||||||
|
|
||||||
while (remainder > 0) {
|
|
||||||
chunk_size = std::min (remainder,
|
|
||||||
std::min ((size_t) (filesize - write_pos),
|
|
||||||
(size_t) (block_size - write_pos % block_size)));
|
|
||||||
|
|
||||||
memcpy (&write_buf [write_pos % block_size], source_ptr, chunk_size);
|
|
||||||
source_ptr += chunk_size;
|
|
||||||
|
|
||||||
write_pos = (write_pos + chunk_size) % filesize;
|
|
||||||
if (write_pos % block_size == 0) {
|
|
||||||
save_write_buf ();
|
|
||||||
write_buf_start_addr = write_pos;
|
|
||||||
|
|
||||||
if (write_buf == read_buf) {
|
|
||||||
if (read_buf == buf2)
|
|
||||||
write_buf = buf1;
|
|
||||||
else
|
|
||||||
write_buf = buf2;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
remainder -= chunk_size;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void zmq::swap_t::fill_buf (char *buf, int64_t pos)
|
|
||||||
{
|
|
||||||
if (file_pos != pos) {
|
|
||||||
#ifdef ZMQ_HAVE_WINDOWS
|
|
||||||
__int64 offset = _lseeki64 (fd, pos, SEEK_SET);
|
|
||||||
#else
|
|
||||||
off_t offset = lseek (fd, (off_t) pos, SEEK_SET);
|
|
||||||
#endif
|
|
||||||
errno_assert (offset == pos);
|
|
||||||
file_pos = pos;
|
|
||||||
}
|
|
||||||
size_t octets_stored = 0;
|
|
||||||
size_t octets_total = std::min (block_size, (size_t) (filesize - file_pos));
|
|
||||||
|
|
||||||
while (octets_stored < octets_total) {
|
|
||||||
#ifdef ZMQ_HAVE_WINDOWS
|
|
||||||
int rc = _read (fd, &buf [octets_stored], octets_total - octets_stored);
|
|
||||||
#else
|
|
||||||
ssize_t rc = read (fd, &buf [octets_stored],
|
|
||||||
octets_total - octets_stored);
|
|
||||||
#endif
|
|
||||||
errno_assert (rc > 0);
|
|
||||||
octets_stored += rc;
|
|
||||||
}
|
|
||||||
file_pos += octets_total;
|
|
||||||
}
|
|
||||||
|
|
||||||
void zmq::swap_t::save_write_buf ()
|
|
||||||
{
|
|
||||||
if (file_pos != write_buf_start_addr) {
|
|
||||||
#ifdef ZMQ_HAVE_WINDOWS
|
|
||||||
__int64 offset = _lseeki64 (fd, write_buf_start_addr, SEEK_SET);
|
|
||||||
#else
|
|
||||||
off_t offset = lseek (fd, (off_t) write_buf_start_addr, SEEK_SET);
|
|
||||||
#endif
|
|
||||||
errno_assert (offset == write_buf_start_addr);
|
|
||||||
file_pos = write_buf_start_addr;
|
|
||||||
}
|
|
||||||
size_t octets_stored = 0;
|
|
||||||
size_t octets_total = std::min (block_size, (size_t) (filesize - file_pos));
|
|
||||||
|
|
||||||
while (octets_stored < octets_total) {
|
|
||||||
#ifdef ZMQ_HAVE_WINDOWS
|
|
||||||
int rc = _write (fd, &write_buf [octets_stored],
|
|
||||||
octets_total - octets_stored);
|
|
||||||
#else
|
|
||||||
ssize_t rc = write (fd, &write_buf [octets_stored],
|
|
||||||
octets_total - octets_stored);
|
|
||||||
#endif
|
|
||||||
errno_assert (rc > 0);
|
|
||||||
octets_stored += rc;
|
|
||||||
}
|
|
||||||
file_pos += octets_total;
|
|
||||||
}
|
|
||||||
|
|
||||||
int64_t zmq::swap_t::buffer_space ()
|
|
||||||
{
|
|
||||||
if (write_pos < read_pos)
|
|
||||||
return read_pos - write_pos;
|
|
||||||
|
|
||||||
return filesize - (write_pos - read_pos);
|
|
||||||
}
|
|
123
src/swap.hpp
123
src/swap.hpp
@ -1,123 +0,0 @@
|
|||||||
/*
|
|
||||||
Copyright (c) 2007-2011 iMatix Corporation
|
|
||||||
Copyright (c) 2007-2011 Other contributors as noted in the AUTHORS file
|
|
||||||
|
|
||||||
This file is part of 0MQ.
|
|
||||||
|
|
||||||
0MQ is free software; you can redistribute it and/or modify it under
|
|
||||||
the terms of the GNU Lesser General Public License as published by
|
|
||||||
the Free Software Foundation; either version 3 of the License, or
|
|
||||||
(at your option) any later version.
|
|
||||||
|
|
||||||
0MQ is distributed in the hope that it will be useful,
|
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
GNU Lesser General Public License for more details.
|
|
||||||
|
|
||||||
You should have received a copy of the GNU Lesser General Public License
|
|
||||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef __ZMQ_SWAP_HPP_INCLUDED__
|
|
||||||
#define __ZMQ_SWAP_HPP_INCLUDED__
|
|
||||||
|
|
||||||
#include "../include/zmq.h"
|
|
||||||
|
|
||||||
#include <string>
|
|
||||||
#include "stdint.hpp"
|
|
||||||
|
|
||||||
namespace zmq
|
|
||||||
{
|
|
||||||
|
|
||||||
// This class implements a message swap. Messages are retrieved from
|
|
||||||
// the swap in the same order as they entered it.
|
|
||||||
|
|
||||||
class swap_t
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
|
|
||||||
enum { default_block_size = 8192 };
|
|
||||||
|
|
||||||
// Creates the swap.
|
|
||||||
swap_t (int64_t filesize_);
|
|
||||||
|
|
||||||
~swap_t ();
|
|
||||||
|
|
||||||
int init ();
|
|
||||||
|
|
||||||
// Stores the message into the swap. The function
|
|
||||||
// returns false if the swap is full; true otherwise.
|
|
||||||
bool store (zmq_msg_t *msg_);
|
|
||||||
|
|
||||||
// Fetches the oldest message from the swap. It is an error
|
|
||||||
// to call this function when the swap is empty.
|
|
||||||
void fetch (zmq_msg_t *msg_);
|
|
||||||
|
|
||||||
void commit ();
|
|
||||||
|
|
||||||
void rollback ();
|
|
||||||
|
|
||||||
// Returns true if the swap is empty; false otherwise.
|
|
||||||
bool empty ();
|
|
||||||
|
|
||||||
|
|
||||||
// // Returns true if and only if the swap is full.
|
|
||||||
// bool full ();
|
|
||||||
|
|
||||||
// Returns true if the message fits into swap.
|
|
||||||
bool fits (zmq_msg_t *msg_);
|
|
||||||
|
|
||||||
private:
|
|
||||||
|
|
||||||
// Copies data from a memory buffer to the backing file.
|
|
||||||
// Wraps around when reaching maximum file size.
|
|
||||||
void copy_from_file (void *buffer_, size_t count_);
|
|
||||||
|
|
||||||
// Copies data from the backing file to the memory buffer.
|
|
||||||
// Wraps around when reaching end-of-file.
|
|
||||||
void copy_to_file (const void *buffer_, size_t count_);
|
|
||||||
|
|
||||||
// Returns the buffer space available.
|
|
||||||
int64_t buffer_space ();
|
|
||||||
|
|
||||||
void fill_buf (char *buf, int64_t pos);
|
|
||||||
|
|
||||||
void save_write_buf ();
|
|
||||||
|
|
||||||
// File descriptor to the backing file.
|
|
||||||
int fd;
|
|
||||||
|
|
||||||
// Name of the backing file.
|
|
||||||
std::string filename;
|
|
||||||
|
|
||||||
// Maximum size of the backing file.
|
|
||||||
int64_t filesize;
|
|
||||||
|
|
||||||
// File offset associated with the fd file descriptor.
|
|
||||||
int64_t file_pos;
|
|
||||||
|
|
||||||
// File offset the next message will be stored at.
|
|
||||||
int64_t write_pos;
|
|
||||||
|
|
||||||
// File offset the next message will be read from.
|
|
||||||
int64_t read_pos;
|
|
||||||
|
|
||||||
int64_t commit_pos;
|
|
||||||
|
|
||||||
size_t block_size;
|
|
||||||
|
|
||||||
char *buf1;
|
|
||||||
char *buf2;
|
|
||||||
char *read_buf;
|
|
||||||
char *write_buf;
|
|
||||||
|
|
||||||
int64_t write_buf_start_addr;
|
|
||||||
|
|
||||||
// Disable copying of the swap object.
|
|
||||||
swap_t (const swap_t&);
|
|
||||||
const swap_t &operator = (const swap_t&);
|
|
||||||
};
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
|
Loading…
Reference in New Issue
Block a user