// -*- c-basic-offset: 4; related-file-name: "../../include/click/standard/scheduleinfo.hh" -*- /* * scheduleinfo.{cc,hh} -- element stores schedule parameters * Benjie Chen, 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 #include #include #include #include #include #include #include CLICK_DECLS ScheduleInfo::ScheduleInfo() { #ifdef HAVE_STRIDE_SCHED static_assert((1 << FRAC_BITS) == Task::DEFAULT_TICKETS); #endif } ScheduleInfo::~ScheduleInfo() { } int ScheduleInfo::configure(Vector &conf, ErrorHandler *errh) { NameDB* db = NameInfo::getdb(NameInfo::T_SCHEDULEINFO, this, 4, true); // compile scheduling info for (int i = 0; i < conf.size(); i++) { Vector parts; int32_t mt; cp_spacevec(conf[i], parts); if (parts.size() == 0) /* empty argument OK */; else if (parts.size() != 2 || !cp_real2(parts[1], FRAC_BITS, &mt)) errh->error("expected 'ELEMENTNAME PARAM', got '%s'", conf[i].c_str()); else db->define(parts[0], &mt, 4); } return 0; } int ScheduleInfo::query(Element *e, ErrorHandler *errh) { #ifdef HAVE_STRIDE_SCHED // check prefixes in order of increasing length String id = e->name(); Vector prefixes; prefixes.push_back(String()); const char *slash = id.begin(); while ((slash = find(slash, id.end(), '/')) < id.end()) { prefixes.push_back(id.substring(id.begin(), slash + 1)); slash++; } prefixes.push_back(id); Vector tickets(prefixes.size(), Task::DEFAULT_TICKETS); NameDB *db = NameInfo::getdb(NameInfo::T_SCHEDULEINFO, e, 4, false); while (db) { bool frobbed = false; for (int i = prefixes.size() - 1; i >= 0 && prefixes[i].length() >= db->prefix().length(); i--) if (db->query(prefixes[i].substring(db->prefix().length()), &tickets[i], 4)) frobbed = true; else if (frobbed) // erase intermediate ticket settings tickets[i] = Task::DEFAULT_TICKETS; db = db->prefix_parent(); } // multiply tickets int tickets_out = tickets[0]; for (int i = 1; i < tickets.size(); i++) if (tickets[i] != Task::DEFAULT_TICKETS) #ifdef HAVE_INT64_TYPES tickets_out = ((int64_t) tickets_out * tickets[i]) >> FRAC_BITS; #else tickets_out = (tickets_out * tickets[i]) >> FRAC_BITS; #endif // check for too many tickets if (tickets_out > Task::MAX_TICKETS) { tickets_out = Task::MAX_TICKETS; String m = cp_unparse_real2(tickets_out, FRAC_BITS); errh->warning("ScheduleInfo too high; reduced to %s", m.c_str()); } // return the result you've got return tickets_out; #else return 1; #endif } void ScheduleInfo::initialize_task(Element *e, Task *task, bool schedule, ErrorHandler *errh) { #ifdef HAVE_STRIDE_SCHED int tickets = query(e, errh); if (tickets > 0) { task->initialize(e, schedule); task->set_tickets(tickets); } #else task->initialize(e, schedule); #endif } CLICK_ENDDECLS EXPORT_ELEMENT(ScheduleInfo) ELEMENT_HEADER()