]> granicus.if.org Git - ipset/commitdiff
Fix checking the revision of the set type at create command
authorJozsef Kadlecsik <kadlec@blackhole.kfki.hu>
Fri, 18 Mar 2011 16:23:43 +0000 (17:23 +0100)
committerJozsef Kadlecsik <kadlec@blackhole.kfki.hu>
Fri, 18 Mar 2011 16:23:43 +0000 (17:23 +0100)
The revision number was not checked at the create command: if the userspace
sent a valid set type but with not supported revision number, it'd create
a loop.

kernel/net/netfilter/ipset/ip_set_core.c

index b4c5600fe44dbba370f4d81f4b93abb0c9e7bd8a..295c80386092f17f750acc69f5f2b6cda5c3e85f 100644 (file)
@@ -94,16 +94,28 @@ static int
 find_set_type_get(const char *name, u8 family, u8 revision,
                  struct ip_set_type **found)
 {
+       struct ip_set_type *type;
+       int err;
+
        rcu_read_lock();
        *found = find_set_type(name, family, revision);
        if (*found) {
-               int err = !try_module_get((*found)->me);
-               rcu_read_unlock();
-               return err ? -EFAULT : 0;
+               err = !try_module_get((*found)->me) ? -EFAULT : 0;
+               goto unlock;
        }
+       /* Make sure the type is loaded but we don't support the revision */
+       list_for_each_entry_rcu(type, &ip_set_type_list, list)
+               if (STREQ(type->name, name)) {
+                       err = -IPSET_ERR_FIND_TYPE;
+                       goto unlock;
+               }
        rcu_read_unlock();
 
        return try_to_load_type(name);
+
+unlock:
+       rcu_read_unlock();
+       return err;
 }
 
 /* Find a given set type by name and family.
@@ -116,7 +128,7 @@ find_set_type_minmax(const char *name, u8 family, u8 *min, u8 *max)
        struct ip_set_type *type;
        bool found = false;
 
-       *min = *max = 0;
+       *min = 255; *max = 0;
        rcu_read_lock();
        list_for_each_entry_rcu(type, &ip_set_type_list, list)
                if (STREQ(type->name, name) &&