// -*- c-basic-offset: 4 -*- /* * fullnotequeue.{cc,hh} -- queue element that notifies on full * Eddie Kohler * * Copyright (c) 2004 Regents of the University of California * * 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 #include "fullnotequeue.hh" CLICK_DECLS FullNoteQueue::FullNoteQueue() { } FullNoteQueue::~FullNoteQueue() { } void * FullNoteQueue::cast(const char *n) { if (strcmp(n, "FullNoteQueue") == 0) return (FullNoteQueue *)this; else if (strcmp(n, Notifier::FULL_NOTIFIER) == 0) return static_cast(&_full_note); else return NotifierQueue::cast(n); } int FullNoteQueue::configure(Vector &conf, ErrorHandler *errh) { _full_note.initialize(router()); _full_note.set_active(true, false); return NotifierQueue::configure(conf, errh); } void FullNoteQueue::push(int, Packet *p) { // Code taken from SimpleQueue::push(). int next = next_i(_tail); if (next != _head) { _q[_tail] = p; _tail = next; int s = size(); if (s > _highwater_length) _highwater_length = s; #if !NOTIFIERQUEUE_LOCK // This can leave a single packet in the queue indefinitely in // multithreaded Click, because of a race condition with pull(). if (!_empty_note.active()) _empty_note.wake(); #else if (s == 1) { _lock.acquire(); if (!_empty_note.active()) _empty_note.wake(); _lock.release(); } #endif if (s == capacity()) _full_note.sleep(); } else { if (_drops == 0) click_chatter("%{element}: overflow", this); _drops++; p->kill(); } } Packet * FullNoteQueue::pull(int) { Packet *p = deq(); if (p) { _sleepiness = 0; if (size() == capacity() - 1) _full_note.wake(); } else if (++_sleepiness == SLEEPINESS_TRIGGER) { #if !NOTIFIERQUEUE_LOCK _empty_note.sleep(); #else _lock.acquire(); if (_head == _tail) // if still empty... _empty_note.sleep(); _lock.release(); #endif } return p; } CLICK_ENDDECLS ELEMENT_REQUIRES(NotifierQueue) EXPORT_ELEMENT(FullNoteQueue FullNoteQueue-FullNoteQueue)