e->list = l;
- LOCK_DOMAIN(event, l->lock);
- if (enlisted(&e->n))
+ struct event_cork *ec = e->cork;
+
+ uint ping = 0;
+
+ if (ec)
{
+ LOCK_DOMAIN(cork, ec->lock);
+ LOCK_DOMAIN(event, l->lock);
+
+ if (!enlisted(&e->n))
+ if (ec->count)
+ add_tail(&ec->events, &e->n);
+ else
+ {
+ add_tail(&l->events, &e->n);
+ ping = 1;
+ }
+
UNLOCK_DOMAIN(event, l->lock);
- return;
+ UNLOCK_DOMAIN(cork, ec->lock);
}
+ else
+ {
+ LOCK_DOMAIN(event, l->lock);
- add_tail(&l->events, &e->n);
- UNLOCK_DOMAIN(event, l->lock);
+ if (!enlisted(&e->n))
+ {
+ add_tail(&l->events, &e->n);
+ ping = 1;
+ }
- birdloop_ping(l->loop);
+ UNLOCK_DOMAIN(event, l->lock);
+ }
+
+ if (ping)
+ birdloop_ping(l->loop);
}
void io_log_event(void *hook, void *data);
return repeat;
}
+
+void ev_cork(struct event_cork *ec)
+{
+ LOCK_DOMAIN(cork, ec->lock);
+ ec->count++;
+ UNLOCK_DOMAIN(cork, ec->lock);
+}
+
+void ev_uncork(struct event_cork *ec)
+{
+ LOCK_DOMAIN(cork, ec->lock);
+
+ if (--ec->count)
+ {
+ UNLOCK_DOMAIN(cork, ec->lock);
+ return;
+ }
+
+ node *n;
+ WALK_LIST_FIRST(n, ec->events)
+ {
+ event *e = SKIP_BACK(event, n, n);
+ event_list *el = e->list;
+
+ rem_node(&e->n);
+
+ LOCK_DOMAIN(event, el->lock);
+ add_tail(&el->events, &e->n);
+ UNLOCK_DOMAIN(event, el->lock);
+
+ birdloop_ping(el->loop);
+ }
+
+ UNLOCK_DOMAIN(cork, ec->lock);
+
+ birdloop_ping(&main_birdloop);
+}
#include <stdatomic.h>
DEFINE_DOMAIN(event);
+DEFINE_DOMAIN(cork);
typedef struct event {
resource r;
void *data;
node n; /* Internal link */
struct event_list *list; /* List where this event is put in */
+ struct event_cork *cork; /* Event execution limiter */
+ node cork_node;
} event;
typedef struct event_list {
DOMAIN(event) lock;
} event_list;
+struct event_cork {
+ DOMAIN(cork) lock;
+ u32 count;
+ list events;
+};
+
extern event_list global_event_list;
extern event_list global_work_list;
el->lock = DOMAIN_NEW(event, name);
}
+static inline void ev_init_cork(struct event_cork *ec, const char *name)
+{
+ init_list(&ec->events);
+ ec->lock = DOMAIN_NEW(cork, name);
+ ec->count = 0;
+};
+
void ev_send(event_list *, event *);
#define ev_send_loop(l, e) ev_send(birdloop_event_list((l)), (e))
#define LEGACY_EVENT_LIST(l) (((l) == &global_event_list) || ((l) == &global_work_list))
+void ev_cork(struct event_cork *);
+void ev_uncork(struct event_cork *);
+
+static inline u32 ev_corked(struct event_cork *ec)
+{
+ if (!ec)
+ return 0;
+
+ LOCK_DOMAIN(cork, ec->lock);
+ u32 out = ec->count;
+ UNLOCK_DOMAIN(cork, ec->lock);
+ return out;
+}
+
_Bool birdloop_inside(struct birdloop *loop);
static inline int
struct domain_generic *the_bird;
struct domain_generic *proto;
struct domain_generic *rtable;
+ struct domain_generic *cork;
struct domain_generic *event;
};
uint fast_rx; /* RX has higher priority in event loop */
uint rbsize;
int (*rx_hook)(struct birdsock *, uint size); /* NULL=receiving turned off, returns 1 to clear rx buffer */
+ struct event_cork *cork; /* Cork to temporarily stop receiving data */
byte *tbuf, *tpos; /* NULL=allocate automatically */
byte *ttx; /* Internal */
}
static inline uint sk_want_events(sock *s)
-{ return (s->rx_hook ? POLLIN : 0) | ((s->ttx != s->tpos) ? POLLOUT : 0); }
+{ return ((s->rx_hook && !ev_corked(s->cork)) ? POLLIN : 0) | ((s->ttx != s->tpos) ? POLLOUT : 0); }
/*
FIXME: this should be called from sock code
{
pfd[nfds] = (struct pollfd) { .fd = -1 }; /* everything other set to 0 by this */
s = SKIP_BACK(sock, n, n);
- if (s->rx_hook)
+ if (s->rx_hook && !ev_corked(s->cork))
{
pfd[nfds].fd = s->fd;
pfd[nfds].events |= POLLIN;