mirror of
https://git.haproxy.org/git/haproxy.git/
synced 2025-08-06 15:17:01 +02:00
Just like task_schedule() but for applets to wakeup an applet at a specific time, leverages _task_schedule() internally
449 lines
13 KiB
C
449 lines
13 KiB
C
/*
|
|
* include/haproxy/applet.h
|
|
* This file contains applet function prototypes
|
|
*
|
|
* Copyright (C) 2000-2015 Willy Tarreau - w@1wt.eu
|
|
*
|
|
* This library 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, version 2.1
|
|
* exclusively.
|
|
*
|
|
* This library 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 library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
#ifndef _HAPROXY_APPLET_H
|
|
#define _HAPROXY_APPLET_H
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <haproxy/api.h>
|
|
#include <haproxy/applet-t.h>
|
|
#include <haproxy/channel.h>
|
|
#include <haproxy/list.h>
|
|
#include <haproxy/pool.h>
|
|
#include <haproxy/sc_strm.h>
|
|
#include <haproxy/session.h>
|
|
#include <haproxy/stconn.h>
|
|
#include <haproxy/task.h>
|
|
|
|
extern unsigned int nb_applets;
|
|
extern struct pool_head *pool_head_appctx;
|
|
|
|
struct task *task_run_applet(struct task *t, void *context, unsigned int state);
|
|
struct task *task_process_applet(struct task *t, void *context, unsigned int state);
|
|
int appctx_buf_available(void *arg);
|
|
void *applet_reserve_svcctx(struct appctx *appctx, size_t size);
|
|
void applet_reset_svcctx(struct appctx *appctx);
|
|
void appctx_shut(struct appctx *appctx);
|
|
|
|
struct appctx *appctx_new_on(struct applet *applet, struct sedesc *sedesc, int thr);
|
|
int appctx_finalize_startup(struct appctx *appctx, struct proxy *px, struct buffer *input);
|
|
void appctx_free_on_early_error(struct appctx *appctx);
|
|
void appctx_free(struct appctx *appctx);
|
|
|
|
size_t appctx_htx_rcv_buf(struct appctx *appctx, struct buffer *buf, size_t count, unsigned int flags);
|
|
size_t appctx_raw_rcv_buf(struct appctx *appctx, struct buffer *buf, size_t count, unsigned int flags);
|
|
size_t appctx_rcv_buf(struct stconn *sc, struct buffer *buf, size_t count, unsigned int flags);
|
|
|
|
size_t appctx_htx_snd_buf(struct appctx *appctx, struct buffer *buf, size_t count, unsigned int flags);
|
|
size_t appctx_raw_snd_buf(struct appctx *appctx, struct buffer *buf, size_t count, unsigned int flags);
|
|
size_t appctx_snd_buf(struct stconn *sc, struct buffer *buf, size_t count, unsigned int flags);
|
|
|
|
int appctx_fastfwd(struct stconn *sc, unsigned int count, unsigned int flags);
|
|
ssize_t applet_append_line(void *ctx, struct ist v1, struct ist v2, size_t ofs, size_t len, char delim);
|
|
static forceinline void applet_fl_set(struct appctx *appctx, uint on);
|
|
static forceinline void applet_fl_clr(struct appctx *appctx, uint off);
|
|
|
|
static inline struct appctx *appctx_new_here(struct applet *applet, struct sedesc *sedesc)
|
|
{
|
|
return appctx_new_on(applet, sedesc, tid);
|
|
}
|
|
|
|
static inline struct appctx *appctx_new_anywhere(struct applet *applet, struct sedesc *sedesc)
|
|
{
|
|
return appctx_new_on(applet, sedesc, -1);
|
|
}
|
|
|
|
|
|
/*
|
|
* Release a buffer, if any, and try to wake up entities waiting in the buffer
|
|
* wait queue.
|
|
*/
|
|
static inline void appctx_release_buf(struct appctx *appctx, struct buffer *bptr)
|
|
{
|
|
if (bptr->size) {
|
|
b_free(bptr);
|
|
offer_buffers(appctx->buffer_wait.target, 1);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Allocate a buffer. If if fails, it adds the appctx in buffer wait queue and
|
|
* sets the relevant blocking flag depending on the side (assuming that bptr is
|
|
* either &appctx->inbuf or &appctx->outbuf). Upon success it will also clear
|
|
* the equivalent MAYALLOC flags.
|
|
*/
|
|
static inline struct buffer *appctx_get_buf(struct appctx *appctx, struct buffer *bptr)
|
|
{
|
|
struct buffer *buf = NULL;
|
|
int is_inbuf = (bptr == &appctx->inbuf);
|
|
|
|
if (likely(!LIST_INLIST(&appctx->buffer_wait.list))) {
|
|
if (unlikely((buf = b_alloc(bptr, is_inbuf ? DB_MUX_TX : DB_SE_RX)) == NULL)) {
|
|
b_queue(is_inbuf ? DB_MUX_TX : DB_SE_RX, &appctx->buffer_wait, appctx, appctx_buf_available);
|
|
applet_fl_set(appctx, is_inbuf ? APPCTX_FL_INBLK_ALLOC : APPCTX_FL_OUTBLK_ALLOC);
|
|
} else {
|
|
applet_fl_clr(appctx, is_inbuf ? APPCTX_FL_IN_MAYALLOC : APPCTX_FL_OUT_MAYALLOC);
|
|
}
|
|
}
|
|
return buf;
|
|
}
|
|
|
|
/* Helper function to call .init applet callback function, if it exists. Returns 0
|
|
* on success and -1 on error.
|
|
*/
|
|
static inline int appctx_init(struct appctx *appctx)
|
|
{
|
|
/* Set appctx affinity to the current thread. Because, after this call,
|
|
* the appctx will be fully initialized. The session and the stream will
|
|
* eventually be created. The affinity must be set now !
|
|
*/
|
|
BUG_ON(appctx->t->tid != tid);
|
|
task_set_thread(appctx->t, tid);
|
|
|
|
if (appctx->applet->init)
|
|
return appctx->applet->init(appctx);
|
|
return 0;
|
|
}
|
|
|
|
/* Releases an appctx previously allocated by appctx_new(). */
|
|
static inline void __appctx_free(struct appctx *appctx)
|
|
{
|
|
appctx_release_buf(appctx, &appctx->inbuf);
|
|
appctx_release_buf(appctx, &appctx->outbuf);
|
|
|
|
task_destroy(appctx->t);
|
|
b_dequeue(&appctx->buffer_wait);
|
|
if (appctx->sess)
|
|
session_free(appctx->sess);
|
|
BUG_ON(appctx->sedesc && !se_fl_test(appctx->sedesc, SE_FL_ORPHAN));
|
|
sedesc_free(appctx->sedesc);
|
|
pool_free(pool_head_appctx, appctx);
|
|
_HA_ATOMIC_DEC(&nb_applets);
|
|
}
|
|
|
|
/* wakes up an applet when conditions have changed. We're using a macro here in
|
|
* order to retrieve the caller's place.
|
|
*/
|
|
#define appctx_wakeup(ctx) \
|
|
_task_wakeup((ctx)->t, TASK_WOKEN_OTHER, MK_CALLER(WAKEUP_TYPE_APPCTX_WAKEUP, 0, 0))
|
|
|
|
#define appctx_schedule(ctx, w) \
|
|
_task_schedule((ctx)->t, w, MK_CALLER(WAKEUP_TYPE_TASK_SCHEDULE, 0, 0))
|
|
|
|
/* returns the stream connector the appctx is attached to, via the sedesc */
|
|
static inline struct stconn *appctx_sc(const struct appctx *appctx)
|
|
{
|
|
return appctx->sedesc->sc;
|
|
}
|
|
|
|
/* returns the stream the appctx is attached to. Note that a stream *must*
|
|
* be attached, as we use an unchecked dereference via __sc_strm().
|
|
*/
|
|
static inline struct stream *appctx_strm(const struct appctx *appctx)
|
|
{
|
|
return __sc_strm(appctx->sedesc->sc);
|
|
}
|
|
|
|
/* returns 1 if the appctx is attached on the backend side or 0 if it is
|
|
* attached on the frontend side. Note that only frontend appctx may have no SC.
|
|
*/
|
|
static inline int appctx_is_back(const struct appctx *appctx)
|
|
{
|
|
struct stconn *sc = appctx_sc(appctx);
|
|
|
|
return !!(sc && (sc->flags & SC_FL_ISBACK));
|
|
}
|
|
|
|
static forceinline void applet_fl_zero(struct appctx *appctx)
|
|
{
|
|
appctx->flags = 0;
|
|
}
|
|
|
|
static forceinline void applet_fl_setall(struct appctx *appctx, uint all)
|
|
{
|
|
appctx->flags = all;
|
|
}
|
|
|
|
static forceinline void applet_fl_set(struct appctx *appctx, uint on)
|
|
{
|
|
if (((on & (APPCTX_FL_EOS|APPCTX_FL_EOI)) && appctx->flags & APPCTX_FL_ERR_PENDING) ||
|
|
((on & APPCTX_FL_ERR_PENDING) && appctx->flags & (APPCTX_FL_EOI|APPCTX_FL_EOS)))
|
|
on |= APPCTX_FL_ERROR;
|
|
appctx->flags |= on;
|
|
}
|
|
|
|
static forceinline void applet_fl_clr(struct appctx *appctx, uint off)
|
|
{
|
|
appctx->flags &= ~off;
|
|
}
|
|
|
|
static forceinline uint applet_fl_test(const struct appctx *appctx, uint test)
|
|
{
|
|
return !!(appctx->flags & test);
|
|
}
|
|
|
|
static forceinline uint applet_fl_get(const struct appctx *appctx)
|
|
{
|
|
return appctx->flags;
|
|
}
|
|
|
|
static inline void applet_set_eoi(struct appctx *appctx)
|
|
{
|
|
applet_fl_set(appctx, APPCTX_FL_EOI);
|
|
}
|
|
|
|
static inline void applet_set_eos(struct appctx *appctx)
|
|
{
|
|
applet_fl_set(appctx, APPCTX_FL_EOS);
|
|
}
|
|
|
|
static inline void applet_set_error(struct appctx *appctx)
|
|
{
|
|
if (applet_fl_test(appctx, (APPCTX_FL_EOS|APPCTX_FL_EOI)))
|
|
applet_fl_set(appctx, APPCTX_FL_ERROR);
|
|
else
|
|
applet_fl_set(appctx, APPCTX_FL_ERR_PENDING);
|
|
}
|
|
|
|
/* The applet announces it has more data to deliver to the stream's input
|
|
* buffer.
|
|
*/
|
|
static inline void applet_have_more_data(struct appctx *appctx)
|
|
{
|
|
se_fl_clr(appctx->sedesc, SE_FL_HAVE_NO_DATA);
|
|
}
|
|
|
|
/* The applet announces it doesn't have more data for the stream's input
|
|
* buffer.
|
|
*/
|
|
static inline void applet_have_no_more_data(struct appctx *appctx)
|
|
{
|
|
se_fl_set(appctx->sedesc, SE_FL_HAVE_NO_DATA);
|
|
}
|
|
|
|
/* The applet indicates that it's ready to consume data from the stream's
|
|
* output buffer. Rely on the corresponding SE function
|
|
*/
|
|
static inline void applet_will_consume(struct appctx *appctx)
|
|
{
|
|
se_will_consume(appctx->sedesc);
|
|
}
|
|
|
|
/* The applet indicates that it's not willing to consume data from the stream's
|
|
* output buffer. Rely on the corresponding SE function
|
|
*/
|
|
static inline void applet_wont_consume(struct appctx *appctx)
|
|
{
|
|
se_wont_consume(appctx->sedesc);
|
|
}
|
|
|
|
/* The applet indicates that it's willing to consume data from the stream's
|
|
* output buffer, but that there's not enough, so it doesn't want to be woken
|
|
* up until more are presented. Rely on the corresponding SE function
|
|
*/
|
|
static inline void applet_need_more_data(struct appctx *appctx)
|
|
{
|
|
se_need_more_data(appctx->sedesc);
|
|
}
|
|
|
|
/* The applet indicates that it does not expect data from the opposite endpoint.
|
|
* This way the stream know it should not trigger read timeout on the other
|
|
* side.
|
|
*/
|
|
static inline void applet_expect_no_data(struct appctx *appctx)
|
|
{
|
|
se_fl_set(appctx->sedesc, SE_FL_EXP_NO_DATA);
|
|
}
|
|
|
|
/* The applet indicates that it expects data from the opposite endpoint. This
|
|
* way the stream know it may trigger read timeout on the other side.
|
|
*/
|
|
static inline void applet_expect_data(struct appctx *appctx)
|
|
{
|
|
se_fl_clr(appctx->sedesc, SE_FL_EXP_NO_DATA);
|
|
}
|
|
|
|
/* Should only be used via wrappers applet_putchk() / applet_putchk_stress(). */
|
|
static inline int _applet_putchk(struct appctx *appctx, struct buffer *chunk,
|
|
int stress)
|
|
{
|
|
int ret;
|
|
|
|
if (appctx->flags & APPCTX_FL_INOUT_BUFS) {
|
|
if (unlikely(stress) ?
|
|
b_data(&appctx->outbuf) :
|
|
b_data(chunk) > b_room(&appctx->outbuf)) {
|
|
applet_fl_set(appctx, APPCTX_FL_OUTBLK_FULL);
|
|
ret = -1;
|
|
}
|
|
else {
|
|
ret = b_putblk(&appctx->outbuf, b_head(chunk), b_data(chunk));
|
|
chunk->data -= ret;
|
|
}
|
|
}
|
|
else {
|
|
struct sedesc *se = appctx->sedesc;
|
|
|
|
if ((unlikely(stress) && ci_data(sc_ic(se->sc))) ||
|
|
(ret = ci_putchk(sc_ic(se->sc), chunk)) < 0) {
|
|
/* XXX: Handle all errors as a lack of space because callers
|
|
* don't handles other cases for now. So applets must be
|
|
* careful to handles shutdown (-2) and invalid calls (-3) by
|
|
* themselves.
|
|
*/
|
|
sc_need_room(se->sc, chunk->data);
|
|
ret = -1;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* writes chunk <chunk> into the input channel of the stream attached to this
|
|
* appctx's endpoint, and marks the SC_FL_NEED_ROOM on a channel full error.
|
|
* See ci_putchk() for the list of return codes.
|
|
*/
|
|
static inline int applet_putchk(struct appctx *appctx, struct buffer *chunk)
|
|
{
|
|
return _applet_putchk(appctx, chunk, 0);
|
|
}
|
|
|
|
/* Equivalent of applet_putchk() but with stress condition alternatives activated. */
|
|
static inline int applet_putchk_stress(struct appctx *appctx, struct buffer *chunk)
|
|
{
|
|
return _applet_putchk(appctx, chunk, 1);
|
|
}
|
|
|
|
/* writes <len> chars from <blk> into the input channel of the stream attached
|
|
* to this appctx's endpoint, and marks the SC_FL_NEED_ROOM on a channel full
|
|
* error. See ci_putblk() for the list of return codes.
|
|
*/
|
|
static inline int applet_putblk(struct appctx *appctx, const char *blk, int len)
|
|
{
|
|
int ret;
|
|
|
|
if (appctx->flags & APPCTX_FL_INOUT_BUFS) {
|
|
if (len > b_room(&appctx->outbuf)) {
|
|
applet_fl_set(appctx, APPCTX_FL_OUTBLK_FULL);
|
|
ret = -1;
|
|
}
|
|
else
|
|
ret = b_putblk(&appctx->outbuf, blk, len);
|
|
}
|
|
else {
|
|
struct sedesc *se = appctx->sedesc;
|
|
|
|
ret = ci_putblk(sc_ic(se->sc), blk, len);
|
|
if (ret < 0) {
|
|
/* XXX: Handle all errors as a lack of space because callers
|
|
* don't handles other cases for now. So applets must be
|
|
* careful to handles shutdown (-2) and invalid calls (-3) by
|
|
* themselves.
|
|
*/
|
|
sc_need_room(se->sc, len);
|
|
ret = -1;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/* writes chars from <str> up to the trailing zero (excluded) into the input
|
|
* channel of the stream attached to this appctx's endpoint, and marks the
|
|
* SC_FL_NEED_ROOM on a channel full error. See ci_putstr() for the list of
|
|
* return codes.
|
|
*/
|
|
static inline int applet_putstr(struct appctx *appctx, const char *str)
|
|
{
|
|
int ret;
|
|
|
|
if (appctx->flags & APPCTX_FL_INOUT_BUFS) {
|
|
int len = strlen(str);
|
|
|
|
if (len > b_room(&appctx->outbuf)) {
|
|
applet_fl_set(appctx, APPCTX_FL_OUTBLK_FULL);
|
|
ret = -1;
|
|
}
|
|
else
|
|
ret = b_putblk(&appctx->outbuf, str, len);
|
|
}
|
|
else {
|
|
struct sedesc *se = appctx->sedesc;
|
|
|
|
ret = ci_putstr(sc_ic(se->sc), str);
|
|
if (ret < 0) {
|
|
/* XXX: Handle all errors as a lack of space because callers
|
|
* don't handles other cases for now. So applets must be
|
|
* careful to handles shutdown (-2) and invalid calls (-3) by
|
|
* themselves.
|
|
*/
|
|
sc_need_room(se->sc, strlen(str));
|
|
ret = -1;
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
/* writes character <chr> into the input channel of the stream attached to this
|
|
* appctx's endpoint, and marks the SC_FL_NEED_ROOM on a channel full error.
|
|
* See ci_putchr() for the list of return codes.
|
|
*/
|
|
static inline int applet_putchr(struct appctx *appctx, char chr)
|
|
{
|
|
int ret;
|
|
|
|
if (appctx->flags & APPCTX_FL_INOUT_BUFS) {
|
|
if (b_full(&appctx->outbuf)) {
|
|
applet_fl_set(appctx, APPCTX_FL_OUTBLK_FULL);
|
|
ret = -1;
|
|
}
|
|
else {
|
|
b_putchr(&appctx->outbuf, chr);
|
|
ret = 1;
|
|
}
|
|
}
|
|
else {
|
|
struct sedesc *se = appctx->sedesc;
|
|
|
|
ret = ci_putchr(sc_ic(se->sc), chr);
|
|
if (ret < 0) {
|
|
/* XXX: Handle all errors as a lack of space because callers
|
|
* don't handles other cases for now. So applets must be
|
|
* careful to handles shutdown (-2) and invalid calls (-3) by
|
|
* themselves.
|
|
*/
|
|
sc_need_room(se->sc, 1);
|
|
ret = -1;
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
#endif /* _HAPROXY_APPLET_H */
|
|
|
|
/*
|
|
* Local variables:
|
|
* c-indent-level: 8
|
|
* c-basic-offset: 8
|
|
* End:
|
|
*/
|