// $Id: SetOfPrefix.cc,v 4.10 2006/06/08 15:16:26 shane Exp $ // Copyright (c) 2001,2002 RIPE NCC // // All Rights Reserved // // Permission to use, copy, modify, and distribute this software and its // documentation for any purpose and without fee is hereby granted, // provided that the above copyright notice appear in all copies and that // both that copyright notice and this permission notice appear in // supporting documentation, and that the name of the author not be // used in advertising or publicity pertaining to distribution of the // software without specific, written prior permission. // // THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING // ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS; IN NO EVENT SHALL // AUTHOR BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY // DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN // AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF // OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. // // // Copyright (c) 1994 by the University of Southern California // All rights reserved. // // Permission to use, copy, modify, and distribute this software and its // documentation in source and binary forms for lawful non-commercial // purposes and without fee is hereby granted, provided that the above // copyright notice appear in all copies and that both the copyright // notice and this permission notice appear in supporting documentation, // and that any documentation, advertising materials, and other materials // related to such distribution and use acknowledge that the software was // developed by the University of Southern California, Information // Sciences Institute. The name of the USC may not be used to endorse or // promote products derived from this software without specific prior // written permission. // // THE UNIVERSITY OF SOUTHERN CALIFORNIA DOES NOT MAKE ANY // REPRESENTATIONS ABOUT THE SUITABILITY OF THIS SOFTWARE FOR ANY // PURPOSE. THIS SOFTWARE IS PROVIDED "AS IS" AND WITHOUT ANY EXPRESS OR // IMPLIED WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED // WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE, // TITLE, AND NON-INFRINGEMENT. // // IN NO EVENT SHALL USC, OR ANY OTHER CONTRIBUTOR BE LIABLE FOR ANY // SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES, WHETHER IN CONTRACT, TORT, // OR OTHER FORM OF ACTION, ARISING OUT OF OR IN CONNECTION WITH, THE USE // OR PERFORMANCE OF THIS SOFTWARE. // // Questions concerning this software should be directed to // ratoolset@isi.edu. // // Author(s): Cengiz Alaettinoglu #pragma implementation #include "config.h" #include "util/debug.hh" #include "SetOfPrefix.hh" #include #include #include "rpsl/prefix.hh" #include using namespace std; void SetOfPrefix::insert(const PrefixRanges& b) { if (_universal) return; u_int64_t rngs; int i; for (int j = b.low(); j < b.fence(); ++j) { // Replaced by wlee@isi.edu for better performance rngs = ~(~(u_int64_t)0 << (33 - b[j].get_n())) & (~(u_int64_t)0 << (32 - b[j].get_m())); /* rngs = 0; for (i = b[j].get_n(); i <= b[j].get_m(); ++i) rngs |= bits[i]; */ if (not_) members.remove(b[j].get_ipaddr(), b[j].get_length(), rngs); else members.insert(b[j].get_ipaddr(), b[j].get_length(), rngs); } } void SetOfPrefix::remove(const PrefixRanges& b) { if (_universal) { _universal = 0; not_ = 1; } u_int64_t rngs; int i; for (int j = b.low(); j < b.fence(); ++j) { // Replaced by wlee@isi.edu for better performance rngs = ~(~(u_int64_t)0 << (33 - b[j].get_n())) & (~(u_int64_t)0 << (32 - b[j].get_m())); /* rngs = 0; for (i = b[j].get_n(); i <= b[j].get_m(); ++i) rngs |= bits[i]; */ if (not_) members.insert(b[j].get_ipaddr(), b[j].get_length(), rngs); else members.remove(b[j].get_ipaddr(), b[j].get_length(), rngs); } } void SetOfPrefix::operator= (const PrefixRanges& b) { clear(); insert(b); } void SetOfPrefix::do_print (ostream& stream) { stream << *this; } ostream& operator<<(ostream& stream, SetOfPrefix& set) { if (set._universal) stream << "UNIVERSAL"; else { if (set.not_) stream << "NOT"; stream << set.members; } return stream; } void SetOfPrefix::operator |= (const SetOfPrefix& b) { // union RadixSet c; if (_universal || b.isEmpty()) ; // done else if (b._universal) { members.clear(); _universal = 1; not_ = 0; } else { switch (not_*2+b.not_) { case 0: // not = 0, b.not = 0 => this UNION b = members UNION b.members members |= b.members; break; case 1: // not = 0, b.not = 1 => this UNION b = members MINUS b.members c = members; members = b.members; members -= c; not_ = 1; break; case 2: // not = 1, b.not = 0 => this UNION b = b.members MINUS members members -= b.members; break; case 3: // not = 1, b.not = 1 => this UNION b = members INTERSECTION b.members members &= b.members; break; default: assert(0); } if (isEmpty() && not_) { // case 1,2,3 can cause this _universal = 1; not_ = 0; } } } void SetOfPrefix::operator &= (const SetOfPrefix& b) { // intersection RadixSet c; if (isEmpty() || b._universal) ; // done else if (b.isEmpty()) { members.clear(); _universal = 0; not_ = 0; } else if (_universal) { _universal = b._universal; not_ = b.not_; members = b.members; } else { switch (not_*2+b.not_) { case 0: // not = 0, b.not = 0 => this INTSCTN b = members INTSCTN b.members members &= b.members; break; case 1: // not = 0, b.not = 1 => this INTSCTN b = members MINUS b.members members -= b.members; break; case 2: // not = 1, b.not = 0 => this INTSCTN b = b.members MINUS members c = members; members = b.members; members -= c; not_ = 0; break; case 3: // not = 1, b.not = 1 => this INTSCTN b = members UNION b.members members |= b.members; break; default: assert(0); } } } void SetOfPrefix::operator ~ () { // complement if (_universal) { assert(members.isEmpty()); _universal = 0; } else if (isEmpty()) { assert(! not_); _universal = 1; } else not_ = ! not_; } int SetOfPrefix::operator == (const SetOfPrefix& other) const { // equivalance return (_universal == other._universal && not_ == other.not_ && members == other.members); } void SetOfPrefix::operator = (const SetOfPrefix& other) { // assignment members.clear(); not_ = other.not_; _universal = other._universal; members = other.members; }