Source: ../../ospf/external.hh


 
LOGO
 Annotated List  Files  Globals  Hierarchy  Index  Top
// -*- c-basic-offset: 4; tab-width: 8; indent-tabs-mode: t -*-
// vim:set sts=4 ts=8:

// Copyright (c) 2001-2009 XORP, Inc.
//
// This program is free software; you can redistribute it and/or modify
// it under the terms of the GNU General Public License, Version 2, June
// 1991 as published by the Free Software Foundation. Redistribution
// and/or modification of this program under the terms of any other
// version of the GNU General Public License is not permitted.
// 
// This program is distributed in the hope that it will be useful, but
// WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. For more details,
// see the GNU General Public License, Version 2, a copy of which can be
// found in the XORP LICENSE.gpl file.
// 
// XORP Inc, 2953 Bunker Hill Lane, Suite 204, Santa Clara, CA 95054, USA;
// http://xorp.net

// $XORP: xorp/ospf/external.hh,v 1.25 2009/01/05 18:31:01 jtc Exp $

#ifndef __OSPF_EXTERNAL_HH__
#define __OSPF_EXTERNAL_HH__

#include <libxorp/trie.hh>

/**
 * Storage for AS-external-LSAs with efficient access.
 */
class ASExternalDatabase {
 public:
    struct compare {
	bool operator ()(const Lsa::LsaRef a, const Lsa::LsaRef b) const {
	    if (a->get_header().get_link_state_id() ==
		b->get_header().get_link_state_id())
		return a->get_header().get_advertising_router() <
		    b->get_header().get_advertising_router();
	    return a->get_header().get_link_state_id() <
		b->get_header().get_link_state_id();
	}
    };

    typedef set <Lsa::LsaRef, compare>::iterator iterator;

    iterator begin() { return _lsas.begin(); }
    iterator end() { return _lsas.end(); }
    void erase(iterator i) { _lsas.erase(i); }
    void insert(Lsa::LsaRef lsar) { _lsas.insert(lsar); }
    void clear();

    iterator find(Lsa::LsaRef lsar);

 private:
    set <Lsa::LsaRef, compare> _lsas;		// Stored AS-external-LSAs.
};

/**
 * Handle AS-external-LSAs.
 */
template <typename A>
class External {
 public:
    External(Ospf<A>& ospf, map<OspfTypes::AreaID, AreaRouter<A> *>& areas);

    /**
     * Candidate for announcing to other areas. Store this LSA for
     * future replay into other areas. Also arrange for the MaxAge
     * timer to start running.
     *
     * @param area the AS-external-LSA came from.
     *
     * @return true if this LSA should be propogated to other areas.
     */
    bool announce(OspfTypes::AreaID area, Lsa::LsaRef lsar);

    /**
     * Called to complete a series of calls to announce().
     */
    bool announce_complete(OspfTypes::AreaID area);

    /**
     * Provide this area with the stored AS-external-LSAs.
     */
    void push(AreaRouter<A> *area_router);

    /**
     * A true external route redistributed from the RIB (announce).
     */
    bool announce(IPNet<A> net, A nexthop, uint32_t metric,
		  const PolicyTags& policytags);

    /**
     * A true external route redistributed from the RIB (withdraw).
     */
    bool withdraw(const IPNet<A>& net);

    /**
     * Clear the AS-external-LSA database.
     */
    bool clear_database();

    /**
     * Suppress or remove AS-external-LSAs that are originated by
     * this router that should yield to externally generated
     * AS-external-LSAs as described in RFC 2328 Section 12.4.4. By
     * time this method is called the routing table should have been
     * updated so it should be clear if the other router is reachable.
     */
    void suppress_lsas(OspfTypes::AreaID area);

    /**
     * A route has just been added to the routing table.
     */
    void suppress_route_announce(OspfTypes::AreaID area, IPNet<A> net,
				 RouteEntry<A>& rt);

    /**
     * A route has just been withdrawn from the routing table.
     */
    void suppress_route_withdraw(OspfTypes::AreaID area, IPNet<A> net,
				 RouteEntry<A>& rt);

    /**
     * Is this an AS boundary router?
     */
    bool as_boundary_router_p() const {
	return _originating != 0;
    }
    
    /**
     * Re-run the policy filters on all routes.
     */
    void push_routes();

 private:
    Ospf<A>& _ospf;			// Reference to the controlling class.
    map<OspfTypes::AreaID, AreaRouter<A> *>& _areas;	// All the areas

    ASExternalDatabase _lsas;		// Stored AS-external-LSAs.
    uint32_t _originating;		// Number of AS-external-LSAs
					// that are currently being originated.
    uint32_t _lsid;			// OSPFv3 only next Link State ID.
    map<IPNet<IPv6>, uint32_t> _lsmap; 	// OSPFv3 only
    list<Lsa::LsaRef> _suppress_temp;	// LSAs that could possibly
					// suppress self originated LSAs
#ifdef SUPPRESS_DB
    Trie<A, Lsa::LsaRef> _suppress_db;	// Database of suppressed self
					// originated LSAs
#endif

    /**
     * Find this LSA
     */
    ASExternalDatabase::iterator find_lsa(Lsa::LsaRef lsar);

    /**
     * Add this LSA to the database if it already exists replace it
     * with this entry.
     */
    void update_lsa(Lsa::LsaRef lsar);

    /**
     * Delete this LSA from the database.
     */
    void delete_lsa(Lsa::LsaRef lsar);

    /**
     * This LSA has reached MaxAge get rid of it from the database and
     * flood it out of all areas.
     */
    void maxage_reached(Lsa::LsaRef lsar);

    /**
     * Networks with same network number but different prefix lengths
     * can generate the same link state ID. When generating a new LSA
     * if a collision occurs use: 
     * RFC 2328 Appendix E. An algorithm for assigning Link State IDs
     * to resolve the clash.
     */
    void unique_link_state_id(Lsa::LsaRef lsar);

    /**
     * Networks with same network number but different prefix lengths
     * can generate the same link state ID. When looking for an LSA
     * make sure that there the lsar that matches the net is found.
     *
     * @param lsar search for this LSA in the database. WARNING this
     * LSA may have its link state ID field modifed by the search.
     * @param net that must match the LSA. 
     */
    ASExternalDatabase::iterator unique_find_lsa(Lsa::LsaRef lsar,
						 const IPNet<A>& net);

    /**
     * Set the network, nexthop and link state ID.
     */
    void set_net_nexthop_lsid(ASExternalLsa *aselsa, IPNet<A> net, A nexthop);


    /**
     * Send this self originated LSA out.
     */
    void announce_lsa(Lsa::LsaRef lsar);

    /**
     * Pass this outbound AS-external-LSA through the policy filter.
     */
    bool do_filtering(IPNet<A>& network, A& nexthop, uint32_t& metric,
		      bool& e_bit, uint32_t& tag, bool& tag_set,
		      const PolicyTags& policytags);

    /**
     * Start the refresh timer.
     */
    void start_refresh_timer(Lsa::LsaRef lsar);

    /**
     * Called every LSRefreshTime seconds to refresh this LSA.
     */
    void refresh(Lsa::LsaRef lsar);

    /**
     * Clone a self orignated LSA that is about to be removed for
     * possible later introduction.
     */
    Lsa::LsaRef clone_lsa(Lsa::LsaRef lsar);

    /**
     * Is this a self originated AS-external-LSA a candidate for suppression?
     */
    bool suppress_candidate(Lsa::LsaRef lsar, IPNet<A> net, A nexthop,
			    uint32_t metric);
      
#ifdef	SUPPRESS_DB
    /**
     * Store a self originated AS-external-LSA that has been
     * suppressed for possible future revival.
     */
    void suppress_database_add(Lsa::LsaRef lsar, const IPNet<A>& net);
    
    /**
     * Delete a self originated AS-external-LSA that is no longer
     * being redistributed.
     */
    void suppress_database_delete(const IPNet<A>& net, bool invalidate);
#endif

    /**
     * Should this AS-external-LSA cause a self originated LSA to be
     * suppressed.
     */
    void suppress_self(Lsa::LsaRef lsar);

    /**
     * Should this AS-external-LSA cause a self originated LSA to be
     * suppressed.
     */
    bool suppress_self_check(Lsa::LsaRef lsar);

    /**
     * Find a self originated AS-external-LSA by network.
     */
    Lsa::LsaRef find_lsa_by_net(IPNet<A> net);

    /**
     * This LSA if its advertising router is reachable matches a self
     * origniated LSA that will be suppressed. Store until the routing
     * computation has completed and the routing table can be checked.
     */
    void suppress_queue_lsa(Lsa::LsaRef lsar);

    /**
     * An AS-external-LSA has reached MaxAge and is being withdrawn.
     * check to see if it was suppressing a self originated LSA.
     */
    void suppress_maxage(Lsa::LsaRef lsar);

    /**
     * If this is an AS-external-LSA that is suppressing a self
     * originated route, then release it.
     */
    void suppress_release_lsa(Lsa::LsaRef lsar);
};

#endif // __OSPF_EXTERNAL_HH__

Generated by: pavlin on kobe.xorp.net on Wed Jan 7 19:11:07 2009, using kdoc 2.0a54+XORP.