2008-04-30 00:09:16 +00:00
|
|
|
/*
|
2009-01-27 22:34:36 +00:00
|
|
|
* Copyright (c) 2008-2009 Niels Provos and Nick Mathewson
|
2008-04-30 00:09:16 +00:00
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
* 3. The name of the author may not be used to endorse or promote products
|
|
|
|
* derived from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
#ifndef _BUFFEREVENT_INTERNAL_H_
|
|
|
|
#define _BUFFEREVENT_INTERNAL_H_
|
|
|
|
|
|
|
|
#ifdef __cplusplus
|
|
|
|
extern "C" {
|
|
|
|
#endif
|
|
|
|
|
2009-01-27 22:30:46 +00:00
|
|
|
#include "event-config.h"
|
2008-04-30 00:09:16 +00:00
|
|
|
#include "evutil.h"
|
2009-04-13 03:08:11 +00:00
|
|
|
#include "defer-internal.h"
|
2009-04-13 03:17:19 +00:00
|
|
|
#include "evthread-internal.h"
|
|
|
|
#include "event2/thread.h"
|
2009-04-13 03:08:11 +00:00
|
|
|
|
|
|
|
struct bufferevent_private {
|
|
|
|
struct bufferevent bev;
|
|
|
|
|
|
|
|
/** Evbuffer callback to enforce watermarks on input. */
|
|
|
|
struct evbuffer_cb_entry *read_watermarks_cb;
|
|
|
|
|
|
|
|
/** If set, read is suspended until evbuffer some. */
|
|
|
|
unsigned read_suspended : 1;
|
2009-04-17 23:12:34 +00:00
|
|
|
/** If set, we should free the lock when we free the bufferevent. */
|
2009-04-13 03:17:19 +00:00
|
|
|
unsigned own_lock : 1;
|
2009-04-13 03:08:11 +00:00
|
|
|
|
2009-04-17 23:12:34 +00:00
|
|
|
unsigned readcb_pending : 1;
|
|
|
|
unsigned writecb_pending : 1;
|
2009-05-05 02:59:26 +00:00
|
|
|
unsigned connecting : 1;
|
2009-04-17 23:12:34 +00:00
|
|
|
short errorcb_pending;
|
|
|
|
int errno_pending;
|
|
|
|
struct deferred_cb deferred;
|
|
|
|
|
2009-04-13 03:08:11 +00:00
|
|
|
enum bufferevent_options options;
|
|
|
|
|
|
|
|
int refcnt;
|
|
|
|
void *lock;
|
|
|
|
};
|
2008-04-30 00:09:16 +00:00
|
|
|
|
2009-02-02 19:22:13 +00:00
|
|
|
/**
|
|
|
|
Implementation table for a bufferevent: holds function pointers and other
|
|
|
|
information to make the various bufferevent types work.
|
|
|
|
*/
|
|
|
|
struct bufferevent_ops {
|
|
|
|
/** The name of the bufferevent's type. */
|
|
|
|
const char *type;
|
|
|
|
/** At what offset into the implementation type will we find a
|
|
|
|
bufferevent structure?
|
2008-04-30 00:09:16 +00:00
|
|
|
|
2009-02-02 19:22:13 +00:00
|
|
|
Example: if the type is implemented as
|
|
|
|
struct bufferevent_x {
|
|
|
|
int extra_data;
|
|
|
|
struct bufferevent bev;
|
|
|
|
}
|
|
|
|
then mem_offset should be offsetof(struct bufferevent_x, bev)
|
|
|
|
*/
|
|
|
|
off_t mem_offset;
|
2008-04-30 00:09:16 +00:00
|
|
|
|
2009-02-02 19:22:13 +00:00
|
|
|
/** Enables one or more of EV_READ|EV_WRITE on a bufferevent. Does
|
|
|
|
not need to adjust the 'enabled' field. Returns 0 on success, -1
|
|
|
|
on failure.
|
|
|
|
*/
|
|
|
|
int (*enable)(struct bufferevent *, short);
|
2008-04-30 00:09:16 +00:00
|
|
|
|
2009-02-02 19:22:13 +00:00
|
|
|
/** Disables one or more of EV_READ|EV_WRITE on a bufferevent. Does
|
|
|
|
not need to adjust the 'enabled' field. Returns 0 on success, -1
|
|
|
|
on failure.
|
|
|
|
*/
|
|
|
|
int (*disable)(struct bufferevent *, short);
|
2008-04-30 00:09:16 +00:00
|
|
|
|
2009-02-02 19:22:13 +00:00
|
|
|
/** Free any storage and deallocate any extra data or structures used
|
|
|
|
in this implementation.
|
|
|
|
*/
|
|
|
|
void (*destruct)(struct bufferevent *);
|
2008-04-30 00:09:16 +00:00
|
|
|
|
2009-02-02 19:22:13 +00:00
|
|
|
/** Called when the timeouts on the bufferevent have changed.*/
|
|
|
|
void (*adj_timeouts)(struct bufferevent *);
|
|
|
|
|
|
|
|
/** Called to flush data. */
|
|
|
|
int (*flush)(struct bufferevent *, short, enum bufferevent_flush_mode);
|
2008-04-30 00:09:16 +00:00
|
|
|
};
|
|
|
|
|
2009-04-10 15:01:31 +00:00
|
|
|
extern const struct bufferevent_ops bufferevent_ops_socket;
|
|
|
|
extern const struct bufferevent_ops bufferevent_ops_filter;
|
|
|
|
extern const struct bufferevent_ops bufferevent_ops_pair;
|
2009-02-02 19:22:13 +00:00
|
|
|
|
|
|
|
/** Initialize the shared parts of a bufferevent. */
|
2009-04-13 03:08:11 +00:00
|
|
|
int bufferevent_init_common(struct bufferevent_private *, struct event_base *, const struct bufferevent_ops *, enum bufferevent_options options);
|
2009-02-02 19:22:13 +00:00
|
|
|
|
|
|
|
/** For internal use: temporarily stop all reads on bufev, because its
|
|
|
|
* read buffer is too full. */
|
|
|
|
void bufferevent_wm_suspend_read(struct bufferevent *bufev);
|
|
|
|
/** For internal use: temporarily stop all reads on bufev, because its
|
|
|
|
* read buffer is too full. */
|
|
|
|
void bufferevent_wm_unsuspend_read(struct bufferevent *bufev);
|
|
|
|
|
2009-04-13 03:17:19 +00:00
|
|
|
int bufferevent_enable_locking(struct bufferevent *bufev, void *lock);
|
2009-04-17 06:57:38 +00:00
|
|
|
void bufferevent_incref(struct bufferevent *bufev);
|
|
|
|
void _bufferevent_decref_and_unlock(struct bufferevent *bufev);
|
2009-04-13 03:17:19 +00:00
|
|
|
|
2009-04-17 23:12:34 +00:00
|
|
|
void _bufferevent_run_readcb(struct bufferevent *bufev);
|
|
|
|
void _bufferevent_run_writecb(struct bufferevent *bufev);
|
|
|
|
void _bufferevent_run_errorcb(struct bufferevent *bufev, short what);
|
|
|
|
|
2009-04-13 03:17:19 +00:00
|
|
|
#define BEV_UPCAST(b) EVUTIL_UPCAST((b), struct bufferevent_private, bev)
|
|
|
|
|
|
|
|
#define BEV_LOCK(b) do { \
|
|
|
|
struct bufferevent_private *locking = BEV_UPCAST(b); \
|
|
|
|
if (locking->lock) \
|
|
|
|
EVLOCK_LOCK(locking->lock, EVTHREAD_WRITE); \
|
|
|
|
} while(0)
|
|
|
|
|
|
|
|
#define BEV_UNLOCK(b) do { \
|
|
|
|
struct bufferevent_private *locking = BEV_UPCAST(b); \
|
|
|
|
if (locking->lock) \
|
|
|
|
EVLOCK_UNLOCK(locking->lock, EVTHREAD_WRITE); \
|
|
|
|
} while(0)
|
|
|
|
|
2008-04-30 00:09:16 +00:00
|
|
|
#ifdef __cplusplus
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif /* _BUFFEREVENT_INTERNAL_H_ */
|