12c593315Sopenharmony_ci/* 22c593315Sopenharmony_ci * nghttp2 - HTTP/2 C Library 32c593315Sopenharmony_ci * 42c593315Sopenharmony_ci * Copyright (c) 2015 Tatsuhiro Tsujikawa 52c593315Sopenharmony_ci * 62c593315Sopenharmony_ci * Permission is hereby granted, free of charge, to any person obtaining 72c593315Sopenharmony_ci * a copy of this software and associated documentation files (the 82c593315Sopenharmony_ci * "Software"), to deal in the Software without restriction, including 92c593315Sopenharmony_ci * without limitation the rights to use, copy, modify, merge, publish, 102c593315Sopenharmony_ci * distribute, sublicense, and/or sell copies of the Software, and to 112c593315Sopenharmony_ci * permit persons to whom the Software is furnished to do so, subject to 122c593315Sopenharmony_ci * the following conditions: 132c593315Sopenharmony_ci * 142c593315Sopenharmony_ci * The above copyright notice and this permission notice shall be 152c593315Sopenharmony_ci * included in all copies or substantial portions of the Software. 162c593315Sopenharmony_ci * 172c593315Sopenharmony_ci * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 182c593315Sopenharmony_ci * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 192c593315Sopenharmony_ci * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 202c593315Sopenharmony_ci * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE 212c593315Sopenharmony_ci * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION 222c593315Sopenharmony_ci * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION 232c593315Sopenharmony_ci * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 242c593315Sopenharmony_ci */ 252c593315Sopenharmony_ci#include "shrpx_rate_limit.h" 262c593315Sopenharmony_ci 272c593315Sopenharmony_ci#include <limits> 282c593315Sopenharmony_ci 292c593315Sopenharmony_ci#include "shrpx_connection.h" 302c593315Sopenharmony_ci#include "shrpx_log.h" 312c593315Sopenharmony_ci 322c593315Sopenharmony_cinamespace shrpx { 332c593315Sopenharmony_ci 342c593315Sopenharmony_cinamespace { 352c593315Sopenharmony_civoid regencb(struct ev_loop *loop, ev_timer *w, int revents) { 362c593315Sopenharmony_ci auto r = static_cast<RateLimit *>(w->data); 372c593315Sopenharmony_ci r->regen(); 382c593315Sopenharmony_ci} 392c593315Sopenharmony_ci} // namespace 402c593315Sopenharmony_ci 412c593315Sopenharmony_ciRateLimit::RateLimit(struct ev_loop *loop, ev_io *w, size_t rate, size_t burst, 422c593315Sopenharmony_ci Connection *conn) 432c593315Sopenharmony_ci : w_(w), 442c593315Sopenharmony_ci loop_(loop), 452c593315Sopenharmony_ci conn_(conn), 462c593315Sopenharmony_ci rate_(rate), 472c593315Sopenharmony_ci burst_(burst), 482c593315Sopenharmony_ci avail_(burst), 492c593315Sopenharmony_ci startw_req_(false) { 502c593315Sopenharmony_ci ev_timer_init(&t_, regencb, 0., 1.); 512c593315Sopenharmony_ci t_.data = this; 522c593315Sopenharmony_ci if (rate_ > 0) { 532c593315Sopenharmony_ci ev_timer_again(loop_, &t_); 542c593315Sopenharmony_ci } 552c593315Sopenharmony_ci} 562c593315Sopenharmony_ci 572c593315Sopenharmony_ciRateLimit::~RateLimit() { ev_timer_stop(loop_, &t_); } 582c593315Sopenharmony_ci 592c593315Sopenharmony_cisize_t RateLimit::avail() const { 602c593315Sopenharmony_ci if (rate_ == 0) { 612c593315Sopenharmony_ci return std::numeric_limits<ssize_t>::max(); 622c593315Sopenharmony_ci } 632c593315Sopenharmony_ci return avail_; 642c593315Sopenharmony_ci} 652c593315Sopenharmony_ci 662c593315Sopenharmony_civoid RateLimit::drain(size_t n) { 672c593315Sopenharmony_ci if (rate_ == 0) { 682c593315Sopenharmony_ci return; 692c593315Sopenharmony_ci } 702c593315Sopenharmony_ci n = std::min(avail_, n); 712c593315Sopenharmony_ci avail_ -= n; 722c593315Sopenharmony_ci if (avail_ == 0) { 732c593315Sopenharmony_ci ev_io_stop(loop_, w_); 742c593315Sopenharmony_ci } 752c593315Sopenharmony_ci} 762c593315Sopenharmony_ci 772c593315Sopenharmony_civoid RateLimit::regen() { 782c593315Sopenharmony_ci if (rate_ == 0) { 792c593315Sopenharmony_ci return; 802c593315Sopenharmony_ci } 812c593315Sopenharmony_ci if (avail_ + rate_ > burst_) { 822c593315Sopenharmony_ci avail_ = burst_; 832c593315Sopenharmony_ci } else { 842c593315Sopenharmony_ci avail_ += rate_; 852c593315Sopenharmony_ci } 862c593315Sopenharmony_ci 872c593315Sopenharmony_ci if (w_->fd >= 0 && avail_ > 0 && startw_req_) { 882c593315Sopenharmony_ci ev_io_start(loop_, w_); 892c593315Sopenharmony_ci handle_tls_pending_read(); 902c593315Sopenharmony_ci } 912c593315Sopenharmony_ci} 922c593315Sopenharmony_ci 932c593315Sopenharmony_civoid RateLimit::startw() { 942c593315Sopenharmony_ci if (w_->fd < 0) { 952c593315Sopenharmony_ci return; 962c593315Sopenharmony_ci } 972c593315Sopenharmony_ci startw_req_ = true; 982c593315Sopenharmony_ci if (rate_ == 0 || avail_ > 0) { 992c593315Sopenharmony_ci ev_io_start(loop_, w_); 1002c593315Sopenharmony_ci handle_tls_pending_read(); 1012c593315Sopenharmony_ci return; 1022c593315Sopenharmony_ci } 1032c593315Sopenharmony_ci} 1042c593315Sopenharmony_ci 1052c593315Sopenharmony_civoid RateLimit::stopw() { 1062c593315Sopenharmony_ci startw_req_ = false; 1072c593315Sopenharmony_ci ev_io_stop(loop_, w_); 1082c593315Sopenharmony_ci} 1092c593315Sopenharmony_ci 1102c593315Sopenharmony_civoid RateLimit::handle_tls_pending_read() { 1112c593315Sopenharmony_ci if (!conn_ || !conn_->tls.ssl || 1122c593315Sopenharmony_ci (SSL_pending(conn_->tls.ssl) == 0 && conn_->tls.rbuf.rleft() == 0 && 1132c593315Sopenharmony_ci (!conn_->tls.initial_handshake_done || 1142c593315Sopenharmony_ci conn_->tls.earlybuf.rleft() == 0))) { 1152c593315Sopenharmony_ci return; 1162c593315Sopenharmony_ci } 1172c593315Sopenharmony_ci 1182c593315Sopenharmony_ci // Note that ev_feed_event works without starting watcher, but we 1192c593315Sopenharmony_ci // only call this function if watcher is active. 1202c593315Sopenharmony_ci ev_feed_event(loop_, w_, EV_READ); 1212c593315Sopenharmony_ci} 1222c593315Sopenharmony_ci 1232c593315Sopenharmony_ci} // namespace shrpx 124