Viewing file: FdSinkChannel.h (5.22 KB) -rw-r--r-- Select action/file-type: (+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
/* * Phusion Passenger - https://www.phusionpassenger.com/ * Copyright (c) 2014-2017 Phusion Holding B.V. * * "Passenger", "Phusion Passenger" and "Union Station" are registered * trademarks of Phusion Holding B.V. * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ #ifndef _PASSENGER_SERVER_KIT_FD_SINK_CHANNEL_H_ #define _PASSENGER_SERVER_KIT_FD_SINK_CHANNEL_H_
#include <oxt/macros.hpp> #include <cerrno> #include <unistd.h> #include <ev.h> #include <jsoncpp/json.h> #include <ServerKit/Channel.h>
namespace Passenger { namespace ServerKit {
using namespace oxt;
class FdSinkChannel: protected Channel { private: ev_io watcher;
static Result _onData(Channel *channel, const MemoryKit::mbuf &buffer, int errcode) { return static_cast<FdSinkChannel *>(channel)->onData(buffer, errcode); }
Result onData(const MemoryKit::mbuf &buffer, int errcode) { if (buffer.size() > 0) { // Data ssize_t ret;
do { ret = ::write(watcher.fd, buffer.start, buffer.size()); } while (ret == -1 && errno == EINTR); if (ret == (ssize_t) buffer.size()) { return Result(ret, false); } else if (ret >= 0) { ev_io_start(ctx->libev->getLoop(), &watcher); stop(); return Result(ret, false); } else if (errno == EAGAIN || errno == EWOULDBLOCK) { ev_io_start(ctx->libev->getLoop(), &watcher); stop(); return Result(0, false); } else { Channel::feedError(errno); return Result(0, false); } } else if (errcode == 0) { // EOF return Channel::Result(0, true); } else { // Error // We do nothing here. The caller is responsible for handling the error. return Channel::Result(0, false); } }
static void _onWritable(EV_P_ ev_io *io, int revents) { FdSinkChannel *self = static_cast<FdSinkChannel *>(io->data); ev_io_stop(self->ctx->libev->getLoop(), &self->watcher); self->start(); }
void initialize() { dataCallback = _onData; watcher.active = false; watcher.fd = -1; watcher.data = this; }
public: FdSinkChannel() { initialize(); }
FdSinkChannel(Context *context) : Channel(context) { initialize(); }
~FdSinkChannel() { if (ctx != NULL && ev_is_active(&watcher)) { ev_io_stop(ctx->libev->getLoop(), &watcher); } }
// May only be called right after construction. OXT_FORCE_INLINE void setContext(Context *context) { Channel::setContext(context); }
void reinitialize(int fd) { Channel::reinitialize(); ev_io_init(&watcher, _onWritable, fd, EV_WRITE); }
void deinitialize() { if (ev_is_active(&watcher)) { ev_io_stop(ctx->libev->getLoop(), &watcher); } watcher.fd = -1; Channel::deinitialize(); }
OXT_FORCE_INLINE int feed(const MemoryKit::mbuf &mbuf) { return Channel::feed(mbuf); }
OXT_FORCE_INLINE int feed(BOOST_RV_REF(MemoryKit::mbuf) mbuf) { return Channel::feed(mbuf); }
OXT_FORCE_INLINE int feedWithoutRefGuard(const MemoryKit::mbuf &mbuf) { return Channel::feedWithoutRefGuard(mbuf); }
OXT_FORCE_INLINE int feedWithoutRefGuard(BOOST_RV_REF(MemoryKit::mbuf) mbuf) { return Channel::feedWithoutRefGuard(mbuf); }
OXT_FORCE_INLINE void feedError(int errcode) { return Channel::feedError(errcode); }
OXT_FORCE_INLINE int getFd() const { return watcher.fd; }
OXT_FORCE_INLINE bool acceptingInput() const { return Channel::acceptingInput(); }
OXT_FORCE_INLINE bool mayAcceptInputLater() const { return Channel::mayAcceptInputLater(); }
OXT_FORCE_INLINE bool hasError() const { return Channel::hasError(); }
OXT_FORCE_INLINE int getErrcode() const { return Channel::getErrcode(); }
OXT_FORCE_INLINE bool ended() const { return Channel::ended(); }
OXT_FORCE_INLINE bool endAcked() const { return Channel::endAcked(); }
OXT_FORCE_INLINE void setConsumedCallback(ConsumedCallback callback) { Channel::consumedCallback = callback; }
OXT_FORCE_INLINE Hooks *getHooks() const { return hooks; }
OXT_FORCE_INLINE void setHooks(Hooks *hooks) { this->hooks = hooks; }
Json::Value inspectAsJson() const { Json::Value doc = Channel::inspectAsJson(); doc["initialized"] = watcher.fd != -1; doc["io_watcher_active"] = (bool) watcher.active; return doc; } };
} // namespace ServerKit } // namespace Passenger
#endif /* _PASSENGER_SERVER_KIT_FD_SINK_CHANNEL_H_ */
|