Viewing file: swoole_coroutine_channel.h (4.18 KB) -rw-r--r-- Select action/file-type: (+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
/* +----------------------------------------------------------------------+ | Swoole | +----------------------------------------------------------------------+ | This source file is subject to version 2.0 of the Apache license, | | that is bundled with this package in the file LICENSE, and is | | available through the world-wide-web at the following url: | | http://www.apache.org/licenses/LICENSE-2.0.html | | If you did not receive a copy of the Apache2.0 license and are unable| | to obtain it through the world-wide-web, please send a note to | | license@swoole.com so we can mail you a copy immediately. | +----------------------------------------------------------------------+ | Author: Tianfeng Han <rango@swoole.com> | | Twosee <twose@qq.com> | +----------------------------------------------------------------------+ */
#pragma once
#include "swoole.h" #include "swoole_coroutine.h"
#include <sys/stat.h>
#include <iostream> #include <string> #include <list> #include <queue>
namespace swoole { namespace coroutine { //------------------------------------------------------------------------------- class Channel { public: enum Opcode { PRODUCER = 1, CONSUMER = 2, };
enum ErrorCode { ERROR_OK = 0, ERROR_TIMEOUT = -1, ERROR_CLOSED = -2, ERROR_CANCELED = -3, };
struct TimeoutMessage { Channel *chan; Opcode type; Coroutine *co; bool error; TimerNode *timer; };
void *pop(double timeout = -1); bool push(void *data, double timeout = -1); bool close();
Channel(size_t _capacity = 1) : capacity(_capacity) {}
~Channel() { if (!producer_queue.empty()) { swoole_error_log(SW_LOG_WARNING, SW_ERROR_CO_HAS_BEEN_DISCARDED, "channel is destroyed, %zu producers will be discarded", producer_queue.size()); } if (!consumer_queue.empty()) { swoole_error_log(SW_LOG_WARNING, SW_ERROR_CO_HAS_BEEN_DISCARDED, "channel is destroyed, %zu consumers will be discarded", consumer_queue.size()); } }
bool is_closed() { return closed; }
bool is_empty() { return data_queue.size() == 0; }
bool is_full() { return data_queue.size() == capacity; }
size_t length() { return data_queue.size(); }
size_t consumer_num() { return consumer_queue.size(); }
size_t producer_num() { return producer_queue.size(); }
void *pop_data() { if (data_queue.size() == 0) { return nullptr; } void *data = data_queue.front(); data_queue.pop(); return data; }
int get_error() { return error_; }
protected: size_t capacity = 1; bool closed = false; int error_ = 0; std::list<Coroutine *> producer_queue; std::list<Coroutine *> consumer_queue; std::queue<void *> data_queue;
static void timer_callback(Timer *timer, TimerNode *tnode);
void yield(enum Opcode type);
void consumer_remove(Coroutine *co) { consumer_queue.remove(co); }
void producer_remove(Coroutine *co) { producer_queue.remove(co); }
Coroutine *pop_coroutine(enum Opcode type) { Coroutine *co; if (type == PRODUCER) { co = producer_queue.front(); producer_queue.pop_front(); swoole_trace_log(SW_TRACE_CHANNEL, "resume producer cid=%ld", co->get_cid()); } else // if (type == CONSUMER) { co = consumer_queue.front(); consumer_queue.pop_front(); swoole_trace_log(SW_TRACE_CHANNEL, "resume consumer cid=%ld", co->get_cid()); } return co; } }; //------------------------------------------------------------------------------- } // namespace coroutine } // namespace swoole
|