summaryrefslogtreecommitdiff
path: root/lib
diff options
context:
space:
mode:
authorStephen Hemminger <shemminger@vyatta.com>2012-07-13 13:53:51 -0700
committerStephen Hemminger <shemminger@vyatta.com>2012-07-13 16:06:31 -0700
commit3c68c870041d5e4e2be560db5ba9e5be08450df8 (patch)
treeb3d27b5a7372738caa02b655199a572b5ea434e1 /lib
parentdefc161a43dfa74b4de8d8289ca099a9d807d9b5 (diff)
downloadvyatta-cfg-qos-3c68c870041d5e4e2be560db5ba9e5be08450df8.tar.gz
vyatta-cfg-qos-3c68c870041d5e4e2be560db5ba9e5be08450df8.zip
Add support for priority-queue policy
Bug 3643 Enhancement to expose kernel priority queuing.
Diffstat (limited to 'lib')
-rw-r--r--lib/Vyatta/Qos/Priority.pm118
-rw-r--r--lib/Vyatta/Qos/ShaperClass.pm5
-rw-r--r--lib/Vyatta/Qos/TrafficShaper.pm4
3 files changed, 125 insertions, 2 deletions
diff --git a/lib/Vyatta/Qos/Priority.pm b/lib/Vyatta/Qos/Priority.pm
new file mode 100644
index 0000000..b148b1c
--- /dev/null
+++ b/lib/Vyatta/Qos/Priority.pm
@@ -0,0 +1,118 @@
+# Priority Queue
+#
+# **** License ****
+# This program is free software; you can redistribute it and/or modify
+# it under the terms of the GNU General Public License version 2 as
+# published by the Free Software Foundation.
+#
+# 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. See the GNU
+# General Public License for more details.
+#
+# This code was originally developed by Vyatta, Inc.
+# Portions created by Vyatta are Copyright (C) 2008 Vyatta, Inc.
+# All Rights Reserved.
+# **** End License ****
+
+package Vyatta::Qos::Priority;
+use strict;
+use warnings;
+
+require Vyatta::Config;
+require Vyatta::Qos::ShaperClass;
+use POSIX;
+
+# Kernel limits on quantum (bytes)
+use constant {
+ MAXQUANTUM => 200000,
+ MINQUANTUM => 1000,
+};
+
+
+# Create a new instance based on config information
+sub new {
+ my ( $that, $config, $name ) = @_;
+ my $level = $config->setLevel();
+ my @classes = _getClasses($level);
+
+ my $self = {};
+ my $class = ref($that) || $that;
+ bless $self, $class;
+
+ $self->{_level} = $level;
+ $self->{_classes} = \@classes;
+ return $self;
+}
+
+sub _getClasses {
+ my $level = shift;
+ my @classes;
+ my $config = new Vyatta::Config;
+
+ $config->setLevel($level);
+ $config->exists("default")
+ or die "$level configuration not complete: missing default class\n";
+
+ $config->setLevel("$level default");
+ push @classes, new Vyatta::Qos::ShaperClass($config);
+ $config->setLevel($level);
+
+ foreach my $id ( $config->listNodes("class") ) {
+ $config->setLevel("$level class $id");
+ push @classes, new Vyatta::Qos::ShaperClass( $config, $id );
+ }
+
+ return @classes;
+}
+
+# Note: priority does not have internal classes, only sub-qdisc's
+#
+# Although Linux supports mapping TOS and priority to bands, this
+# is not used here. Instead we apply filters to traffic and statically
+# assign bands.
+sub commands {
+ my ( $self, $dev ) = @_;
+ my $classes = $self->{_classes};
+ my $default = shift @$classes;
+ my $maxid = 1;
+ my $bands = 2;
+
+ foreach my $class (@$classes) {
+ my $level = "$self->{_level} class $class->{id}";
+ $class->valid_leaf( $level );
+
+ # find largest class id
+ $maxid = $class->{id}
+ if ( defined $class->{id} && $class->{id} > $maxid );
+
+ # find largest priority
+ my $prio = $class->{_priority};
+
+ $bands = $prio + 1
+ if ( defined $prio && $prio >= $bands);
+ }
+
+ # fill in id of default
+ $default->{id} = ++$maxid;
+ unshift @$classes, $default;
+
+ my $parent = 1;
+ my $root = "root";
+
+ # Since we use filters to set priority
+ printf "qdisc add dev %s %s handle %x: prio bands %d\n",
+ $dev, $root, $parent, $bands;
+
+ # prio is not really classful!
+ foreach my $class (@$classes) {
+ $class->gen_leaf( $dev, $parent );
+
+ my $prio = 1;
+ foreach my $match ( $class->matchRules() ) {
+ $match->filter( $dev, $parent, $class->{id}, $prio++ );
+ }
+ }
+}
+
+1;
diff --git a/lib/Vyatta/Qos/ShaperClass.pm b/lib/Vyatta/Qos/ShaperClass.pm
index 4afd90c..1cb7fbe 100644
--- a/lib/Vyatta/Qos/ShaperClass.pm
+++ b/lib/Vyatta/Qos/ShaperClass.pm
@@ -234,6 +234,10 @@ sub rateCheck {
$rate / 1000;
exit 1;
}
+}
+
+sub valid_leaf {
+ my ($self, $level, $rate) = @_;
my $qtype = $self->{_qdisc};
my $q = $qdiscValidate{$qtype};
@@ -268,7 +272,6 @@ sub gen_class {
print " quantum $quantum" if ($quantum);
print " burst $self->{_burst}" if ( $self->{_burst} );
print " prio $self->{_priority}" if ( $self->{_priority} );
-
print "\n";
}
diff --git a/lib/Vyatta/Qos/TrafficShaper.pm b/lib/Vyatta/Qos/TrafficShaper.pm
index 3fd687a..5e365c5 100644
--- a/lib/Vyatta/Qos/TrafficShaper.pm
+++ b/lib/Vyatta/Qos/TrafficShaper.pm
@@ -144,7 +144,9 @@ sub commands {
$default->rateCheck( $rate, "$self->{_level} default" );
foreach my $class (@$classes) {
- $class->rateCheck( $rate, "$self->{_level} class $class->{id}" );
+ my $level = "$self->{_level} class $class->{id}";
+ $class->rateCheck( $rate, $level );
+ $class->valid_leaf( $level );
# find largest class id
if ( defined $class->{id} && $class->{id} > $maxid ) {