blob: f2805be373a1fb5866e3f88e97a146ac246a703e [file]
/* Copyright 2015 greenbytes GmbH (https://www.greenbytes.de)
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef __mod_h2__h2_mplx__
#define __mod_h2__h2_mplx__
/**
* The stream multiplexer. It pushes buckets from the connection
* thread to the stream task threads and vice versa. It's thread-safe
* to use.
*
* There is one h2_mplx instance for each h2_session, which sits on top
* of a particular httpd conn_rec. Input goes from the connection to
* the stream tasks. Output goes from the stream tasks to the connection,
* e.g. the client.
*
* For each stream, there can be at most "H2StreamMaxMemSize" output bytes
* queued in the multiplexer. If a task thread tries to write more
* data, it is blocked until space becomes available.
*
* Writing input is never blocked. In order to use flow control on the input,
* the mplx can be polled for input data consumption.
*/
struct apr_pool_t;
struct apr_thread_mutex_t;
struct apr_thread_cond_t;
struct h2_config;
struct h2_response;
struct h2_task;
struct h2_stream;
struct h2_request;
struct h2_io_set;
struct apr_thread_cond_t;
struct h2_worker;
struct h2_workers;
struct h2_stream_set;
struct h2_task_queue;
#include "h2_io.h"
typedef struct h2_mplx h2_mplx;
/**
* Callback invoked for every stream that had input data read since
* the last invocation.
*/
typedef void h2_mplx_consumed_cb(void *ctx, int stream_id, apr_off_t consumed);
struct h2_mplx {
long id;
APR_RING_ENTRY(h2_mplx) link;
volatile int refs;
conn_rec *c;
apr_pool_t *pool;
apr_bucket_alloc_t *bucket_alloc;
struct h2_task_queue *q;
struct h2_io_set *stream_ios;
struct h2_io_set *ready_ios;
apr_thread_mutex_t *lock;
struct apr_thread_cond_t *added_output;
struct apr_thread_cond_t *join_wait;
int aborted;
apr_size_t stream_max_mem;
apr_pool_t *spare_pool; /* spare pool, ready for next io */
struct h2_workers *workers;
int file_handles_allowed;
h2_mplx_consumed_cb *input_consumed;
void *input_consumed_ctx;
};
/*******************************************************************************
* Object lifecycle and information.
******************************************************************************/
/**
* Create the multiplexer for the given HTTP2 session.
* Implicitly has reference count 1.
*/
h2_mplx *h2_mplx_create(conn_rec *c, apr_pool_t *master,
const struct h2_config *conf,
struct h2_workers *workers);
/**
* Increase the reference counter of this mplx.
*/
void h2_mplx_reference(h2_mplx *m);
/**
* Decreases the reference counter of this mplx.
*/
void h2_mplx_release(h2_mplx *m);
/**
* Decreases the reference counter of this mplx and waits for it
* to reached 0, destroy the mplx afterwards.
* This is to be called from the thread that created the mplx in
* the first place.
* @param m the mplx to be released and destroyed
* @param wait condition var to wait on for ref counter == 0
*/
apr_status_t h2_mplx_release_and_join(h2_mplx *m, struct apr_thread_cond_t *wait);
/**
* Aborts the multiplexer. It will answer all future invocation with
* APR_ECONNABORTED, leading to early termination of ongoing tasks.
*/
void h2_mplx_abort(h2_mplx *mplx);
void h2_mplx_task_done(h2_mplx *m, int stream_id);
/*******************************************************************************
* IO lifetime of streams.
******************************************************************************/
/**
* Notifies mplx that a stream has finished processing.
*
* @param m the mplx itself
* @param stream_id the id of the stream being done
* @param rst_error if != 0, the stream was reset with the error given
*
*/
apr_status_t h2_mplx_stream_done(h2_mplx *m, int stream_id, int rst_error);
/* Return != 0 iff the multiplexer has data for the given stream.
*/
int h2_mplx_out_has_data_for(h2_mplx *m, int stream_id);
/**
* Waits on output data from any stream in this session to become available.
* Returns APR_TIMEUP if no data arrived in the given time.
*/
apr_status_t h2_mplx_out_trywait(h2_mplx *m, apr_interval_time_t timeout,
struct apr_thread_cond_t *iowait);
/*******************************************************************************
* Stream processing.
******************************************************************************/
/**
* Process a stream request.
*
* @param m the multiplexer
* @param stream_id the identifier of the stream
* @param r the request to be processed
* @param eos if input is complete
* @param cmp the stream priority compare function
* @param ctx context data for the compare function
*/
apr_status_t h2_mplx_process(h2_mplx *m, int stream_id,
const struct h2_request *r, int eos,
h2_stream_pri_cmp *cmp, void *ctx);
/**
* Stream priorities have changed, reschedule pending tasks.
*
* @param m the multiplexer
* @param cmp the stream priority compare function
* @param ctx context data for the compare function
*/
apr_status_t h2_mplx_reprioritize(h2_mplx *m, h2_stream_pri_cmp *cmp, void *ctx);
struct h2_task *h2_mplx_pop_task(h2_mplx *mplx, struct h2_worker *w, int *has_more);
/**
* Register a callback for the amount of input data consumed per stream. The
* will only ever be invoked from the thread creating this h2_mplx, e.g. when
* calls from that thread into this h2_mplx are made.
*
* @param m the multiplexer to register the callback at
* @param cb the function to invoke
* @param ctx user supplied argument to invocation.
*/
void h2_mplx_set_consumed_cb(h2_mplx *m, h2_mplx_consumed_cb *cb, void *ctx);
/*******************************************************************************
* Input handling of streams.
******************************************************************************/
/**
* Reads a buckets for the given stream_id. Will return ARP_EAGAIN when
* called with APR_NONBLOCK_READ and no data present. Will return APR_EOF
* when the end of the stream input has been reached.
* The condition passed in will be used for blocking/signalling and will
* be protected by the mplx's own mutex.
*/
apr_status_t h2_mplx_in_read(h2_mplx *m, apr_read_type_e block,
int stream_id, apr_bucket_brigade *bb,
struct apr_thread_cond_t *iowait);
/**
* Appends data to the input of the given stream. Storage of input data is
* not subject to flow control.
*/
apr_status_t h2_mplx_in_write(h2_mplx *mplx, int stream_id,
apr_bucket_brigade *bb);
/**
* Closes the input for the given stream_id.
*/
apr_status_t h2_mplx_in_close(h2_mplx *m, int stream_id);
/**
* Returns != 0 iff the input for the given stream has been closed. There
* could still be data queued, but it can be read without blocking.
*/
int h2_mplx_in_has_eos_for(h2_mplx *m, int stream_id);
/**
* Invoke the consumed callback for all streams that had bytes read since the
* last call to this function. If no stream had input data consumed, the
* callback is not invoked.
* The consumed callback may also be invoked at other times whenever
* the need arises.
* Returns APR_SUCCESS when an update happened, APR_EAGAIN if no update
* happened.
*/
apr_status_t h2_mplx_in_update_windows(h2_mplx *m);
/*******************************************************************************
* Output handling of streams.
******************************************************************************/
/**
* Get a stream whose response is ready for submit. Will set response and
* any out data available in stream.
* @param m the mplxer to get a response from
* @param bb the brigade to place any existing repsonse body data into
*/
struct h2_stream *h2_mplx_next_submit(h2_mplx *m,
struct h2_stream_set *streams);
/**
* Reads output data from the given stream. Will never block, but
* return APR_EAGAIN until data arrives or the stream is closed.
*/
apr_status_t h2_mplx_out_readx(h2_mplx *mplx, int stream_id,
h2_io_data_cb *cb, void *ctx,
apr_off_t *plen, int *peos,
apr_table_t **ptrailers);
/**
* Reads output data into the given brigade. Will never block, but
* return APR_EAGAIN until data arrives or the stream is closed.
*/
apr_status_t h2_mplx_out_read_to(h2_mplx *mplx, int stream_id,
apr_bucket_brigade *bb,
apr_off_t *plen, int *peos,
apr_table_t **ptrailers);
/**
* Opens the output for the given stream with the specified response.
*/
apr_status_t h2_mplx_out_open(h2_mplx *mplx, int stream_id,
struct h2_response *response,
ap_filter_t* filter, apr_bucket_brigade *bb,
struct apr_thread_cond_t *iowait);
/**
* Append the brigade to the stream output. Might block if amount
* of bytes buffered reaches configured max.
* @param stream_id the stream identifier
* @param filter the apache filter context of the data
* @param bb the bucket brigade to append
* @param trailers optional trailers for response, maybe NULL
* @param iowait a conditional used for block/signalling in h2_mplx
*/
apr_status_t h2_mplx_out_write(h2_mplx *mplx, int stream_id,
ap_filter_t* filter, apr_bucket_brigade *bb,
apr_table_t *trailers,
struct apr_thread_cond_t *iowait);
/**
* Closes the output for stream stream_id. Optionally forwards trailers
* fromt the processed stream.
*/
apr_status_t h2_mplx_out_close(h2_mplx *m, int stream_id, apr_table_t *trailers);
apr_status_t h2_mplx_out_rst(h2_mplx *m, int stream_id, int error);
/*******************************************************************************
* h2_mplx list Manipulation.
******************************************************************************/
/**
* The magic pointer value that indicates the head of a h2_mplx list
* @param b The mplx list
* @return The magic pointer value
*/
#define H2_MPLX_LIST_SENTINEL(b) APR_RING_SENTINEL((b), h2_mplx, link)
/**
* Determine if the mplx list is empty
* @param b The list to check
* @return true or false
*/
#define H2_MPLX_LIST_EMPTY(b) APR_RING_EMPTY((b), h2_mplx, link)
/**
* Return the first mplx in a list
* @param b The list to query
* @return The first mplx in the list
*/
#define H2_MPLX_LIST_FIRST(b) APR_RING_FIRST(b)
/**
* Return the last mplx in a list
* @param b The list to query
* @return The last mplx int he list
*/
#define H2_MPLX_LIST_LAST(b) APR_RING_LAST(b)
/**
* Insert a single mplx at the front of a list
* @param b The list to add to
* @param e The mplx to insert
*/
#define H2_MPLX_LIST_INSERT_HEAD(b, e) do { \
h2_mplx *ap__b = (e); \
APR_RING_INSERT_HEAD((b), ap__b, h2_mplx, link); \
} while (0)
/**
* Insert a single mplx at the end of a list
* @param b The list to add to
* @param e The mplx to insert
*/
#define H2_MPLX_LIST_INSERT_TAIL(b, e) do { \
h2_mplx *ap__b = (e); \
APR_RING_INSERT_TAIL((b), ap__b, h2_mplx, link); \
} while (0)
/**
* Get the next mplx in the list
* @param e The current mplx
* @return The next mplx
*/
#define H2_MPLX_NEXT(e) APR_RING_NEXT((e), link)
/**
* Get the previous mplx in the list
* @param e The current mplx
* @return The previous mplx
*/
#define H2_MPLX_PREV(e) APR_RING_PREV((e), link)
/**
* Remove a mplx from its list
* @param e The mplx to remove
*/
#define H2_MPLX_REMOVE(e) APR_RING_REMOVE((e), link)
#endif /* defined(__mod_h2__h2_mplx__) */