cannam@149: // Copyright (c) 2013-2014 Sandstorm Development Group, Inc. and contributors cannam@149: // Licensed under the MIT License: cannam@149: // cannam@149: // Permission is hereby granted, free of charge, to any person obtaining a copy cannam@149: // of this software and associated documentation files (the "Software"), to deal cannam@149: // in the Software without restriction, including without limitation the rights cannam@149: // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell cannam@149: // copies of the Software, and to permit persons to whom the Software is cannam@149: // furnished to do so, subject to the following conditions: cannam@149: // cannam@149: // The above copyright notice and this permission notice shall be included in cannam@149: // all copies or substantial portions of the Software. cannam@149: // cannam@149: // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR cannam@149: // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, cannam@149: // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE cannam@149: // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER cannam@149: // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, cannam@149: // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN cannam@149: // THE SOFTWARE. cannam@149: cannam@149: #ifndef KJ_ASYNC_IO_H_ cannam@149: #define KJ_ASYNC_IO_H_ cannam@149: cannam@149: #if defined(__GNUC__) && !KJ_HEADER_WARNINGS cannam@149: #pragma GCC system_header cannam@149: #endif cannam@149: cannam@149: #include "async.h" cannam@149: #include "function.h" cannam@149: #include "thread.h" cannam@149: #include "time.h" cannam@149: cannam@149: struct sockaddr; cannam@149: cannam@149: namespace kj { cannam@149: cannam@149: #if _WIN32 cannam@149: class Win32EventPort; cannam@149: #else cannam@149: class UnixEventPort; cannam@149: #endif cannam@149: cannam@149: class NetworkAddress; cannam@149: class AsyncOutputStream; cannam@149: cannam@149: // ======================================================================================= cannam@149: // Streaming I/O cannam@149: cannam@149: class AsyncInputStream { cannam@149: // Asynchronous equivalent of InputStream (from io.h). cannam@149: cannam@149: public: cannam@149: virtual Promise read(void* buffer, size_t minBytes, size_t maxBytes); cannam@149: virtual Promise tryRead(void* buffer, size_t minBytes, size_t maxBytes) = 0; cannam@149: cannam@149: Promise read(void* buffer, size_t bytes); cannam@149: cannam@149: virtual Maybe tryGetLength(); cannam@149: // Get the remaining number of bytes that will be produced by this stream, if known. cannam@149: // cannam@149: // This is used e.g. to fill in the Content-Length header of an HTTP message. If unknown, the cannam@149: // HTTP implementation may need to fall back to Transfer-Encoding: chunked. cannam@149: // cannam@149: // The default implementation always returns null. cannam@149: cannam@149: virtual Promise pumpTo( cannam@149: AsyncOutputStream& output, uint64_t amount = kj::maxValue); cannam@149: // Read `amount` bytes from this stream (or to EOF) and write them to `output`, returning the cannam@149: // total bytes actually pumped (which is only less than `amount` if EOF was reached). cannam@149: // cannam@149: // Override this if your stream type knows how to pump itself to certain kinds of output cannam@149: // streams more efficiently than via the naive approach. You can use cannam@149: // kj::dynamicDowncastIfAvailable() to test for stream types you recognize, and if none match, cannam@149: // delegate to the default implementation. cannam@149: // cannam@149: // The default implementation first tries calling output.tryPumpFrom(), but if that fails, it cannam@149: // performs a naive pump by allocating a buffer and reading to it / writing from it in a loop. cannam@149: cannam@149: Promise> readAllBytes(); cannam@149: Promise readAllText(); cannam@149: // Read until EOF and return as one big byte array or string. cannam@149: }; cannam@149: cannam@149: class AsyncOutputStream { cannam@149: // Asynchronous equivalent of OutputStream (from io.h). cannam@149: cannam@149: public: cannam@149: virtual Promise write(const void* buffer, size_t size) = 0; cannam@149: virtual Promise write(ArrayPtr> pieces) = 0; cannam@149: cannam@149: virtual Maybe> tryPumpFrom( cannam@149: AsyncInputStream& input, uint64_t amount = kj::maxValue); cannam@149: // Implements double-dispatch for AsyncInputStream::pumpTo(). cannam@149: // cannam@149: // This method should only be called from within an implementation of pumpTo(). cannam@149: // cannam@149: // This method examines the type of `input` to find optimized ways to pump data from it to this cannam@149: // output stream. If it finds one, it performs the pump. Otherwise, it returns null. cannam@149: // cannam@149: // The default implementation always returns null. cannam@149: }; cannam@149: cannam@149: class AsyncIoStream: public AsyncInputStream, public AsyncOutputStream { cannam@149: // A combination input and output stream. cannam@149: cannam@149: public: cannam@149: virtual void shutdownWrite() = 0; cannam@149: // Cleanly shut down just the write end of the stream, while keeping the read end open. cannam@149: cannam@149: virtual void abortRead() {} cannam@149: // Similar to shutdownWrite, but this will shut down the read end of the stream, and should only cannam@149: // be called when an error has occurred. cannam@149: cannam@149: virtual void getsockopt(int level, int option, void* value, uint* length); cannam@149: virtual void setsockopt(int level, int option, const void* value, uint length); cannam@149: // Corresponds to getsockopt() and setsockopt() syscalls. Will throw an "unimplemented" exception cannam@149: // if the stream is not a socket or the option is not appropriate for the socket type. The cannam@149: // default implementations always throw "unimplemented". cannam@149: cannam@149: virtual void getsockname(struct sockaddr* addr, uint* length); cannam@149: virtual void getpeername(struct sockaddr* addr, uint* length); cannam@149: // Corresponds to getsockname() and getpeername() syscalls. Will throw an "unimplemented" cannam@149: // exception if the stream is not a socket. The default implementations always throw cannam@149: // "unimplemented". cannam@149: // cannam@149: // Note that we don't provide methods that return NetworkAddress because it usually wouldn't cannam@149: // be useful. You can't connect() to or listen() on these addresses, obviously, because they are cannam@149: // ephemeral addresses for a single connection. cannam@149: }; cannam@149: cannam@149: struct OneWayPipe { cannam@149: // A data pipe with an input end and an output end. (Typically backed by pipe() system call.) cannam@149: cannam@149: Own in; cannam@149: Own out; cannam@149: }; cannam@149: cannam@149: struct TwoWayPipe { cannam@149: // A data pipe that supports sending in both directions. Each end's output sends data to the cannam@149: // other end's input. (Typically backed by socketpair() system call.) cannam@149: cannam@149: Own ends[2]; cannam@149: }; cannam@149: cannam@149: class ConnectionReceiver { cannam@149: // Represents a server socket listening on a port. cannam@149: cannam@149: public: cannam@149: virtual Promise> accept() = 0; cannam@149: // Accept the next incoming connection. cannam@149: cannam@149: virtual uint getPort() = 0; cannam@149: // Gets the port number, if applicable (i.e. if listening on IP). This is useful if you didn't cannam@149: // specify a port when constructing the NetworkAddress -- one will have been assigned cannam@149: // automatically. cannam@149: cannam@149: virtual void getsockopt(int level, int option, void* value, uint* length); cannam@149: virtual void setsockopt(int level, int option, const void* value, uint length); cannam@149: // Same as the methods of AsyncIoStream. cannam@149: }; cannam@149: cannam@149: // ======================================================================================= cannam@149: // Datagram I/O cannam@149: cannam@149: class AncillaryMessage { cannam@149: // Represents an ancillary message (aka control message) received using the recvmsg() system cannam@149: // call (or equivalent). Most apps will not use this. cannam@149: cannam@149: public: cannam@149: inline AncillaryMessage(int level, int type, ArrayPtr data); cannam@149: AncillaryMessage() = default; cannam@149: cannam@149: inline int getLevel() const; cannam@149: // Originating protocol / socket level. cannam@149: cannam@149: inline int getType() const; cannam@149: // Protocol-specific message type. cannam@149: cannam@149: template cannam@149: inline Maybe as(); cannam@149: // Interpret the ancillary message as the given struct type. Most ancillary messages are some cannam@149: // sort of struct, so this is a convenient way to access it. Returns nullptr if the message cannam@149: // is smaller than the struct -- this can happen if the message was truncated due to cannam@149: // insufficient ancillary buffer space. cannam@149: cannam@149: template cannam@149: inline ArrayPtr asArray(); cannam@149: // Interpret the ancillary message as an array of items. If the message size does not evenly cannam@149: // divide into elements of type T, the remainder is discarded -- this can happen if the message cannam@149: // was truncated due to insufficient ancillary buffer space. cannam@149: cannam@149: private: cannam@149: int level; cannam@149: int type; cannam@149: ArrayPtr data; cannam@149: // Message data. In most cases you should use `as()` or `asArray()`. cannam@149: }; cannam@149: cannam@149: class DatagramReceiver { cannam@149: // Class encapsulating the recvmsg() system call. You must specify the DatagramReceiver's cannam@149: // capacity in advance; if a received packet is larger than the capacity, it will be truncated. cannam@149: cannam@149: public: cannam@149: virtual Promise receive() = 0; cannam@149: // Receive a new message, overwriting this object's content. cannam@149: // cannam@149: // receive() may reuse the same buffers for content and ancillary data with each call. cannam@149: cannam@149: template cannam@149: struct MaybeTruncated { cannam@149: T value; cannam@149: cannam@149: bool isTruncated; cannam@149: // True if the Receiver's capacity was insufficient to receive the value and therefore the cannam@149: // value is truncated. cannam@149: }; cannam@149: cannam@149: virtual MaybeTruncated> getContent() = 0; cannam@149: // Get the content of the datagram. cannam@149: cannam@149: virtual MaybeTruncated> getAncillary() = 0; cannam@149: // Ancilarry messages received with the datagram. See the recvmsg() system call and the cmsghdr cannam@149: // struct. Most apps don't need this. cannam@149: // cannam@149: // If the returned value is truncated, then the last message in the array may itself be cannam@149: // truncated, meaning its as() method will return nullptr or its asArray() method will cannam@149: // return fewer elements than expected. Truncation can also mean that additional messages were cannam@149: // available but discarded. cannam@149: cannam@149: virtual NetworkAddress& getSource() = 0; cannam@149: // Get the datagram sender's address. cannam@149: cannam@149: struct Capacity { cannam@149: size_t content = 8192; cannam@149: // How much space to allocate for the datagram content. If a datagram is received that is cannam@149: // larger than this, it will be truncated, with no way to recover the tail. cannam@149: cannam@149: size_t ancillary = 0; cannam@149: // How much space to allocate for ancillary messages. As with content, if the ancillary data cannam@149: // is larger than this, it will be truncated. cannam@149: }; cannam@149: }; cannam@149: cannam@149: class DatagramPort { cannam@149: public: cannam@149: virtual Promise send(const void* buffer, size_t size, NetworkAddress& destination) = 0; cannam@149: virtual Promise send(ArrayPtr> pieces, cannam@149: NetworkAddress& destination) = 0; cannam@149: cannam@149: virtual Own makeReceiver( cannam@149: DatagramReceiver::Capacity capacity = DatagramReceiver::Capacity()) = 0; cannam@149: // Create a new `Receiver` that can be used to receive datagrams. `capacity` specifies how much cannam@149: // space to allocate for the received message. The `DatagramPort` must outlive the `Receiver`. cannam@149: cannam@149: virtual uint getPort() = 0; cannam@149: // Gets the port number, if applicable (i.e. if listening on IP). This is useful if you didn't cannam@149: // specify a port when constructing the NetworkAddress -- one will have been assigned cannam@149: // automatically. cannam@149: cannam@149: virtual void getsockopt(int level, int option, void* value, uint* length); cannam@149: virtual void setsockopt(int level, int option, const void* value, uint length); cannam@149: // Same as the methods of AsyncIoStream. cannam@149: }; cannam@149: cannam@149: // ======================================================================================= cannam@149: // Networks cannam@149: cannam@149: class NetworkAddress { cannam@149: // Represents a remote address to which the application can connect. cannam@149: cannam@149: public: cannam@149: virtual Promise> connect() = 0; cannam@149: // Make a new connection to this address. cannam@149: // cannam@149: // The address must not be a wildcard ("*"). If it is an IP address, it must have a port number. cannam@149: cannam@149: virtual Own listen() = 0; cannam@149: // Listen for incoming connections on this address. cannam@149: // cannam@149: // The address must be local. cannam@149: cannam@149: virtual Own bindDatagramPort(); cannam@149: // Open this address as a datagram (e.g. UDP) port. cannam@149: // cannam@149: // The address must be local. cannam@149: cannam@149: virtual Own clone() = 0; cannam@149: // Returns an equivalent copy of this NetworkAddress. cannam@149: cannam@149: virtual String toString() = 0; cannam@149: // Produce a human-readable string which hopefully can be passed to Network::parseAddress() cannam@149: // to reproduce this address, although whether or not that works of course depends on the Network cannam@149: // implementation. This should be called only to display the address to human users, who will cannam@149: // hopefully know what they are able to do with it. cannam@149: }; cannam@149: cannam@149: class Network { cannam@149: // Factory for NetworkAddress instances, representing the network services offered by the cannam@149: // operating system. cannam@149: // cannam@149: // This interface typically represents broad authority, and well-designed code should limit its cannam@149: // use to high-level startup code and user interaction. Low-level APIs should accept cannam@149: // NetworkAddress instances directly and work from there, if at all possible. cannam@149: cannam@149: public: cannam@149: virtual Promise> parseAddress(StringPtr addr, uint portHint = 0) = 0; cannam@149: // Construct a network address from a user-provided string. The format of the address cannam@149: // strings is not specified at the API level, and application code should make no assumptions cannam@149: // about them. These strings should always be provided by humans, and said humans will know cannam@149: // what format to use in their particular context. cannam@149: // cannam@149: // `portHint`, if provided, specifies the "standard" IP port number for the application-level cannam@149: // service in play. If the address turns out to be an IP address (v4 or v6), and it lacks a cannam@149: // port number, this port will be used. If `addr` lacks a port number *and* `portHint` is cannam@149: // omitted, then the returned address will only support listen() and bindDatagramPort() cannam@149: // (not connect()), and an unused port will be chosen each time one of those methods is called. cannam@149: cannam@149: virtual Own getSockaddr(const void* sockaddr, uint len) = 0; cannam@149: // Construct a network address from a legacy struct sockaddr. cannam@149: }; cannam@149: cannam@149: // ======================================================================================= cannam@149: // I/O Provider cannam@149: cannam@149: class AsyncIoProvider { cannam@149: // Class which constructs asynchronous wrappers around the operating system's I/O facilities. cannam@149: // cannam@149: // Generally, the implementation of this interface must integrate closely with a particular cannam@149: // `EventLoop` implementation. Typically, the EventLoop implementation itself will provide cannam@149: // an AsyncIoProvider. cannam@149: cannam@149: public: cannam@149: virtual OneWayPipe newOneWayPipe() = 0; cannam@149: // Creates an input/output stream pair representing the ends of a one-way pipe (e.g. created with cannam@149: // the pipe(2) system call). cannam@149: cannam@149: virtual TwoWayPipe newTwoWayPipe() = 0; cannam@149: // Creates two AsyncIoStreams representing the two ends of a two-way pipe (e.g. created with cannam@149: // socketpair(2) system call). Data written to one end can be read from the other. cannam@149: cannam@149: virtual Network& getNetwork() = 0; cannam@149: // Creates a new `Network` instance representing the networks exposed by the operating system. cannam@149: // cannam@149: // DO NOT CALL THIS except at the highest levels of your code, ideally in the main() function. If cannam@149: // you call this from low-level code, then you are preventing higher-level code from injecting an cannam@149: // alternative implementation. Instead, if your code needs to use network functionality, it cannam@149: // should ask for a `Network` as a constructor or method parameter, so that higher-level code can cannam@149: // chose what implementation to use. The system network is essentially a singleton. See: cannam@149: // http://www.object-oriented-security.org/lets-argue/singletons cannam@149: // cannam@149: // Code that uses the system network should not make any assumptions about what kinds of cannam@149: // addresses it will parse, as this could differ across platforms. String addresses should come cannam@149: // strictly from the user, who will know how to write them correctly for their system. cannam@149: // cannam@149: // With that said, KJ currently supports the following string address formats: cannam@149: // - IPv4: "1.2.3.4", "1.2.3.4:80" cannam@149: // - IPv6: "1234:5678::abcd", "[1234:5678::abcd]:80" cannam@149: // - Local IP wildcard (covers both v4 and v6): "*", "*:80" cannam@149: // - Symbolic names: "example.com", "example.com:80", "example.com:http", "1.2.3.4:http" cannam@149: // - Unix domain: "unix:/path/to/socket" cannam@149: cannam@149: struct PipeThread { cannam@149: // A combination of a thread and a two-way pipe that communicates with that thread. cannam@149: // cannam@149: // The fields are intentionally ordered so that the pipe will be destroyed (and therefore cannam@149: // disconnected) before the thread is destroyed (and therefore joined). Thus if the thread cannam@149: // arranges to exit when it detects disconnect, destruction should be clean. cannam@149: cannam@149: Own thread; cannam@149: Own pipe; cannam@149: }; cannam@149: cannam@149: virtual PipeThread newPipeThread( cannam@149: Function startFunc) = 0; cannam@149: // Create a new thread and set up a two-way pipe (socketpair) which can be used to communicate cannam@149: // with it. One end of the pipe is passed to the thread's start function and the other end of cannam@149: // the pipe is returned. The new thread also gets its own `AsyncIoProvider` instance and will cannam@149: // already have an active `EventLoop` when `startFunc` is called. cannam@149: // cannam@149: // TODO(someday): I'm not entirely comfortable with this interface. It seems to be doing too cannam@149: // much at once but I'm not sure how to cleanly break it down. cannam@149: cannam@149: virtual Timer& getTimer() = 0; cannam@149: // Returns a `Timer` based on real time. Time does not pass while event handlers are running -- cannam@149: // it only updates when the event loop polls for system events. This means that calling `now()` cannam@149: // on this timer does not require a system call. cannam@149: // cannam@149: // This timer is not affected by changes to the system date. It is unspecified whether the timer cannam@149: // continues to count while the system is suspended. cannam@149: }; cannam@149: cannam@149: class LowLevelAsyncIoProvider { cannam@149: // Similar to `AsyncIoProvider`, but represents a lower-level interface that may differ on cannam@149: // different operating systems. You should prefer to use `AsyncIoProvider` over this interface cannam@149: // whenever possible, as `AsyncIoProvider` is portable and friendlier to dependency-injection. cannam@149: // cannam@149: // On Unix, this interface can be used to import native file descriptors into the async framework. cannam@149: // Different implementations of this interface might work on top of different event handling cannam@149: // primitives, such as poll vs. epoll vs. kqueue vs. some higher-level event library. cannam@149: // cannam@149: // On Windows, this interface can be used to import native HANDLEs into the async framework. cannam@149: // Different implementations of this interface might work on top of different event handling cannam@149: // primitives, such as I/O completion ports vs. completion routines. cannam@149: // cannam@149: // TODO(port): Actually implement Windows support. cannam@149: cannam@149: public: cannam@149: // --------------------------------------------------------------------------- cannam@149: // Unix-specific stuff cannam@149: cannam@149: enum Flags { cannam@149: // Flags controlling how to wrap a file descriptor. cannam@149: cannam@149: TAKE_OWNERSHIP = 1 << 0, cannam@149: // The returned object should own the file descriptor, automatically closing it when destroyed. cannam@149: // The close-on-exec flag will be set on the descriptor if it is not already. cannam@149: // cannam@149: // If this flag is not used, then the file descriptor is not automatically closed and the cannam@149: // close-on-exec flag is not modified. cannam@149: cannam@149: #if !_WIN32 cannam@149: ALREADY_CLOEXEC = 1 << 1, cannam@149: // Indicates that the close-on-exec flag is known already to be set, so need not be set again. cannam@149: // Only relevant when combined with TAKE_OWNERSHIP. cannam@149: // cannam@149: // On Linux, all system calls which yield new file descriptors have flags or variants which cannam@149: // set the close-on-exec flag immediately. Unfortunately, other OS's do not. cannam@149: cannam@149: ALREADY_NONBLOCK = 1 << 2 cannam@149: // Indicates that the file descriptor is known already to be in non-blocking mode, so the flag cannam@149: // need not be set again. Otherwise, all wrap*Fd() methods will enable non-blocking mode cannam@149: // automatically. cannam@149: // cannam@149: // On Linux, all system calls which yield new file descriptors have flags or variants which cannam@149: // enable non-blocking mode immediately. Unfortunately, other OS's do not. cannam@149: #endif cannam@149: }; cannam@149: cannam@149: #if _WIN32 cannam@149: typedef uintptr_t Fd; cannam@149: // On Windows, the `fd` parameter to each of these methods must be a SOCKET, and must have the cannam@149: // flag WSA_FLAG_OVERLAPPED (which socket() uses by default, but WSASocket() wants you to specify cannam@149: // explicitly). cannam@149: #else cannam@149: typedef int Fd; cannam@149: // On Unix, any arbitrary file descriptor is supported. cannam@149: #endif cannam@149: cannam@149: virtual Own wrapInputFd(Fd fd, uint flags = 0) = 0; cannam@149: // Create an AsyncInputStream wrapping a file descriptor. cannam@149: // cannam@149: // `flags` is a bitwise-OR of the values of the `Flags` enum. cannam@149: cannam@149: virtual Own wrapOutputFd(Fd fd, uint flags = 0) = 0; cannam@149: // Create an AsyncOutputStream wrapping a file descriptor. cannam@149: // cannam@149: // `flags` is a bitwise-OR of the values of the `Flags` enum. cannam@149: cannam@149: virtual Own wrapSocketFd(Fd fd, uint flags = 0) = 0; cannam@149: // Create an AsyncIoStream wrapping a socket file descriptor. cannam@149: // cannam@149: // `flags` is a bitwise-OR of the values of the `Flags` enum. cannam@149: cannam@149: virtual Promise> wrapConnectingSocketFd( cannam@149: Fd fd, const struct sockaddr* addr, uint addrlen, uint flags = 0) = 0; cannam@149: // Create an AsyncIoStream wrapping a socket and initiate a connection to the given address. cannam@149: // The returned promise does not resolve until connection has completed. cannam@149: // cannam@149: // `flags` is a bitwise-OR of the values of the `Flags` enum. cannam@149: cannam@149: virtual Own wrapListenSocketFd(Fd fd, uint flags = 0) = 0; cannam@149: // Create an AsyncIoStream wrapping a listen socket file descriptor. This socket should already cannam@149: // have had `bind()` and `listen()` called on it, so it's ready for `accept()`. cannam@149: // cannam@149: // `flags` is a bitwise-OR of the values of the `Flags` enum. cannam@149: cannam@149: virtual Own wrapDatagramSocketFd(Fd fd, uint flags = 0); cannam@149: cannam@149: virtual Timer& getTimer() = 0; cannam@149: // Returns a `Timer` based on real time. Time does not pass while event handlers are running -- cannam@149: // it only updates when the event loop polls for system events. This means that calling `now()` cannam@149: // on this timer does not require a system call. cannam@149: // cannam@149: // This timer is not affected by changes to the system date. It is unspecified whether the timer cannam@149: // continues to count while the system is suspended. cannam@149: }; cannam@149: cannam@149: Own newAsyncIoProvider(LowLevelAsyncIoProvider& lowLevel); cannam@149: // Make a new AsyncIoProvider wrapping a `LowLevelAsyncIoProvider`. cannam@149: cannam@149: struct AsyncIoContext { cannam@149: Own lowLevelProvider; cannam@149: Own provider; cannam@149: WaitScope& waitScope; cannam@149: cannam@149: #if _WIN32 cannam@149: Win32EventPort& win32EventPort; cannam@149: #else cannam@149: UnixEventPort& unixEventPort; cannam@149: // TEMPORARY: Direct access to underlying UnixEventPort, mainly for waiting on signals. This cannam@149: // field will go away at some point when we have a chance to improve these interfaces. cannam@149: #endif cannam@149: }; cannam@149: cannam@149: AsyncIoContext setupAsyncIo(); cannam@149: // Convenience method which sets up the current thread with everything it needs to do async I/O. cannam@149: // The returned objects contain an `EventLoop` which is wrapping an appropriate `EventPort` for cannam@149: // doing I/O on the host system, so everything is ready for the thread to start making async calls cannam@149: // and waiting on promises. cannam@149: // cannam@149: // You would typically call this in your main() loop or in the start function of a thread. cannam@149: // Example: cannam@149: // cannam@149: // int main() { cannam@149: // auto ioContext = kj::setupAsyncIo(); cannam@149: // cannam@149: // // Now we can call an async function. cannam@149: // Promise textPromise = getHttp(*ioContext.provider, "http://example.com"); cannam@149: // cannam@149: // // And we can wait for the promise to complete. Note that you can only use `wait()` cannam@149: // // from the top level, not from inside a promise callback. cannam@149: // String text = textPromise.wait(ioContext.waitScope); cannam@149: // print(text); cannam@149: // return 0; cannam@149: // } cannam@149: // cannam@149: // WARNING: An AsyncIoContext can only be used in the thread and process that created it. In cannam@149: // particular, note that after a fork(), an AsyncIoContext created in the parent process will cannam@149: // not work correctly in the child, even if the parent ceases to use its copy. In particular cannam@149: // note that this means that server processes which daemonize themselves at startup must wait cannam@149: // until after daemonization to create an AsyncIoContext. cannam@149: cannam@149: // ======================================================================================= cannam@149: // inline implementation details cannam@149: cannam@149: inline AncillaryMessage::AncillaryMessage( cannam@149: int level, int type, ArrayPtr data) cannam@149: : level(level), type(type), data(data) {} cannam@149: cannam@149: inline int AncillaryMessage::getLevel() const { return level; } cannam@149: inline int AncillaryMessage::getType() const { return type; } cannam@149: cannam@149: template cannam@149: inline Maybe AncillaryMessage::as() { cannam@149: if (data.size() >= sizeof(T)) { cannam@149: return *reinterpret_cast(data.begin()); cannam@149: } else { cannam@149: return nullptr; cannam@149: } cannam@149: } cannam@149: cannam@149: template cannam@149: inline ArrayPtr AncillaryMessage::asArray() { cannam@149: return arrayPtr(reinterpret_cast(data.begin()), data.size() / sizeof(T)); cannam@149: } cannam@149: cannam@149: } // namespace kj cannam@149: cannam@149: #endif // KJ_ASYNC_IO_H_