📄 frontdropqueue.cc
字号:
// -*- c-basic-offset: 4 -*-/* * frontdropqueue.{cc,hh} -- queue element that drops front when full * Eddie Kohler * * Copyright (c) 1999-2000 Massachusetts Institute of Technology * * 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, subject to the conditions * listed in the Click LICENSE file. These conditions include: you must * preserve this copyright notice, and you cannot mention the copyright * holders in advertising related to the Software without their permission. * The Software is provided WITHOUT ANY WARRANTY, EXPRESS OR IMPLIED. This * notice is a summary of the Click LICENSE file; the license in that file is * legally binding. */#include <click/config.h>#include "frontdropqueue.hh"#include <click/confparse.hh>#include <click/error.hh>CLICK_DECLSFrontDropQueue::FrontDropQueue(){}FrontDropQueue::~FrontDropQueue(){}void *FrontDropQueue::cast(const char *n){ if (strcmp(n, "FrontDropQueue") == 0) return (FrontDropQueue *)this; else return NotifierQueue::cast(n);}intFrontDropQueue::live_reconfigure(Vector<String> &conf, ErrorHandler *errh){ // change the maximum queue length at runtime int old_capacity = _capacity; if (configure(conf, errh) < 0) return -1; if (_capacity == old_capacity) return 0; int new_capacity = _capacity; _capacity = old_capacity; Packet **new_q = new Packet *[new_capacity + 1]; if (new_q == 0) return errh->error("out of memory"); int i, j; for (i = _tail - 1, j = new_capacity; i != _head; i = prev_i(i)) { new_q[--j] = _q[i]; if (j == 0) break; } for (; i != _head; i = prev_i(i)) _q[i]->kill(); delete[] _q; _q = new_q; _head = j; _tail = new_capacity; _capacity = new_capacity; return 0;}voidFrontDropQueue::take_state(Element *e, ErrorHandler *errh){ SimpleQueue *q = (SimpleQueue *)e->cast("SimpleQueue"); if (!q) return; if (_tail != _head || _head != 0) { errh->error("already have packets enqueued, can't take state"); return; } _tail = _capacity; int i = _capacity, j = q->tail(); while (i > 0 && j != q->head()) { i--; j = q->prev_i(j); _q[i] = q->packet(j); } _head = i; _highwater_length = size(); if (j != q->head()) errh->warning("some packets lost (old length %d, new capacity %d)", q->size(), _capacity); while (j != q->head()) { j = q->prev_i(j); q->packet(j)->kill(); } q->set_head(0); q->set_tail(0);}voidFrontDropQueue::push(int, Packet *p){ assert(p); // inline Queue::enq() for speed int next = next_i(_tail); // should this stuff be in Queue::enq? if (next == _head) { if (_drops == 0) click_chatter("%{element}: overflow", this); _q[_head]->kill(); _drops++; _head = next_i(_head); } _q[_tail] = p; _tail = next; int s = size(); if (s > _highwater_length) _highwater_length = s; if (s == 1 && !_empty_note.signal_active()) _empty_note.wake_listeners();}CLICK_ENDDECLSELEMENT_REQUIRES(NotifierQueue)EXPORT_ELEMENT(FrontDropQueue)
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -