📄 elementt.cc
字号:
// -*- c-basic-offset: 4 -*-/* * elementt.{cc,hh} -- tool definition of element * Eddie Kohler * * Copyright (c) 1999-2000 Massachusetts Institute of Technology * Copyright (c) 2000 Mazu Networks, Inc. * Copyright (c) 2001 International Computer Science Institute * Copyright (c) 2007 Regents of the University of California * Copyright (c) 2009 Meraki, Inc. * * 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 "elementt.hh"#include "eclasst.hh"#include "routert.hh"#include <click/straccum.hh>#include <click/confparse.hh>#include <click/variableenv.hh>#include <stdlib.h>ElementT::ElementT() : flags(0), _eindex(-1), _type(0), _resolved_type(0), _resolved_type_status(0), _tunnel_input(0), _tunnel_output(0), _owner(0), _user_data(0){}ElementT::ElementT(const String &n, ElementClassT *eclass, const String &config, const LandmarkT &lm) : flags(0), _eindex(-1), _name(n), _type(eclass), _resolved_type(0), _resolved_type_status(0), _configuration(config), _landmark(lm), _ninputs(0), _noutputs(0), _tunnel_input(0), _tunnel_output(0), _owner(0), _user_data(0){ assert(_type); assert(name_ok(_name, true)); _type->use();}ElementT::ElementT(const ElementT &o) : flags(o.flags), _eindex(-1), _name(o._name), _type(o._type), _resolved_type(o._resolved_type), _resolved_type_status(o._resolved_type_status), _configuration(o._configuration), _landmark(o._landmark), _ninputs(0), _noutputs(0), _tunnel_input(0), _tunnel_output(0), _owner(0), _user_data(o._user_data){ if (_type) _type->use(); if (_resolved_type) _resolved_type->use();}ElementT::~ElementT(){ if (_type) _type->unuse(); if (_resolved_type) _resolved_type->unuse();}voidElementT::set_type(ElementClassT *t){ assert(t); t->use(); if (_type) _type->unuse(); _type = t; unresolve_type();}voidElementT::full_kill(){ if (_type) { if (_owner) { RouterT::conn_iterator ci; while ((ci = _owner->begin_connections_from(this)) != _owner->end_connections()) _owner->kill_connection(ci); while ((ci = _owner->begin_connections_to(this)) != _owner->end_connections()) _owner->kill_connection(ci); } _type->unuse(); _type = 0; unresolve_type(); }}boolElementT::name_ok(const String &name, bool allow_anon_names){ const char *data = name.data(); int pos = 0, len = name.length(); // check anonymous name syntax if (len > 0 && data[pos] == ';' && allow_anon_names) { pos++; int epos = len - 1; while (epos > 1 && isdigit((unsigned char) data[epos])) epos--; if (epos == len - 1 || data[epos] != '@') return false; } // must have at least one character, must not start with slash if (pos >= len || data[pos] == '/') return false; while (1) { if (isdigit((unsigned char) data[pos])) { // all-digit component? while (pos < len && isdigit((unsigned char) data[pos])) pos++; if (pos >= len || data[pos] == '/') return false; } while (pos < len && (isalnum((unsigned char) data[pos]) || data[pos] == '_' || data[pos] == '@')) pos++; if (pos == len) return true; else if (data[pos] != '/' || pos == len - 1 || data[pos + 1] == '/') return false; else pos++; }}voidElementT::redeclaration_error(ErrorHandler *errh, const char *what, String name, const String &landmark, const String &old_landmark){ if (!what) what = ""; const char *sp = (strlen(what) ? " " : ""); errh->lerror(landmark, "redeclaration of %s%s%<%s%>", what, sp, name.c_str()); errh->lerror(old_landmark, "%<%s%> previously declared here", name.c_str());}ElementClassT *ElementT::resolve(const VariableEnvironment &env, VariableEnvironment *new_env, ErrorHandler *errh) const{ if (!_type) return 0; // primitives get no scope, no point in expanding if (_type->primitive()) return _type; // expand configuration and do full resolve Vector<String> conf; cp_argvec(cp_expand(_configuration, env), conf); ElementClassT *t = _type->resolve(_ninputs, _noutputs, conf, errh ? errh : ErrorHandler::silent_handler(), _landmark); if (!t) t = _type; if (new_env) t->create_scope(conf, env, *new_env); return t;}ElementClassT *ElementT::resolved_type(const VariableEnvironment &env, ErrorHandler *errh) const{ if (!_type) return 0; if (_resolved_type && !(_resolved_type_status & resolved_type_fragile) && (!(_resolved_type_status & resolved_type_error) || !errh)) return _resolved_type; _resolved_type_status = 0; if (!_type->need_resolve()) { assert(!_resolved_type); _resolved_type = _type; _resolved_type->use(); return _resolved_type; } errh = (errh ? errh : ErrorHandler::silent_handler()); int before = errh->nerrors(); ElementClassT *t = resolve(env, 0, errh); if (errh->nerrors() != before) _resolved_type_status |= resolved_type_error; if (_type->overloaded()) _resolved_type_status |= resolved_type_fragile; if (_resolved_type) _resolved_type->unuse(); _resolved_type = t; _resolved_type->use(); return t;}const PortT PortT::null_port;intPortT::index_in(const Vector<PortT> &v, int start) const{ int size = v.size(); for (int i = start; i < size; i++) if (v[i] == *this) return i; return -1;}intPortT::force_index_in(Vector<PortT> &v, int start) const{ int size = v.size(); for (int i = start; i < size; i++) if (v[i] == *this) return i; v.push_back(*this); return size;}extern "C" {static intPortT_sorter(const void *av, const void *bv){ const PortT *a = (const PortT *)av, *b = (const PortT *)bv; if (a->element == b->element) return a->port - b->port; else return a->element->eindex() - b->element->eindex();}}voidPortT::sort(Vector<PortT> &v){ qsort(&v[0], v.size(), sizeof(PortT), PortT_sorter);}StringPortT::unparse(bool isoutput) const{ if (!element) return String::make_stable("<>"); else if (isoutput) return element->name() + "[" + String(port) + "]"; else return "[" + String(port) + "]" + element->name();}ConnectionT::ConnectionT(const PortT &from, const PortT &to, const LandmarkT &lm) : _landmark(lm){ _end[end_to] = to; _end[end_from] = from; _next[0] = _next[1] = -1;}ConnectionT::ConnectionT(const PortT &from, const PortT &to, const LandmarkT &lm, int next_from, int next_to) : _landmark(lm){ _end[end_to] = to; _end[end_from] = from; _next[end_to] = next_to; _next[end_from] = next_from;}StringConnectionT::unparse() const{ return from().unparse_output() + " -> " + to().unparse_input();}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -