]> git.ipfire.org Git - thirdparty/libnftnl.git/commitdiff
expr: objref: add support for stateful object maps
authorPablo Neira Ayuso <pablo@netfilter.org>
Sun, 27 Nov 2016 22:27:11 +0000 (23:27 +0100)
committerPablo Neira Ayuso <pablo@netfilter.org>
Fri, 9 Dec 2016 13:50:53 +0000 (14:50 +0100)
If the NFT_SET_OBJECT flag is set, then this set stores a mapping
between any random user-defined arbitrary key and one stateful object.
Very useful for performance lookups.

Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org>
include/libnftnl/expr.h
src/expr/objref.c

index f6ea69dfacae10531308091511e36bd68a4ed61d..ee0784928550882660f354b27c4045d415a2b02b 100644 (file)
@@ -237,6 +237,9 @@ enum {
 enum {
        NFTNL_EXPR_OBJREF_IMM_TYPE      = NFTNL_EXPR_BASE,
        NFTNL_EXPR_OBJREF_IMM_NAME,
+       NFTNL_EXPR_OBJREF_SET_SREG,
+       NFTNL_EXPR_OBJREF_SET_NAME,
+       NFTNL_EXPR_OBJREF_SET_ID,
 };
 
 /*
index 01bea7b8b4b185f42134614f971441963cbdbd71..4cfa3cbfbd2b570cb02030b2a6350adeebc40a5a 100644 (file)
@@ -25,6 +25,11 @@ struct nftnl_expr_objref {
                uint32_t        type;
                const char      *name;
        } imm;
+       struct {
+               uint32_t        sreg;
+               const char      *name;
+               uint32_t        id;
+       } set;
 };
 
 static int nftnl_expr_objref_set(struct nftnl_expr *e, uint16_t type,
@@ -41,6 +46,17 @@ static int nftnl_expr_objref_set(struct nftnl_expr *e, uint16_t type,
                if (!objref->imm.name)
                        return -1;
                break;
+       case NFTNL_EXPR_OBJREF_SET_SREG:
+               objref->set.sreg = *((uint32_t *)data);
+               break;
+       case NFTNL_EXPR_OBJREF_SET_NAME:
+               objref->set.name = strdup(data);
+               if (!objref->set.name)
+                       return -1;
+               break;
+       case NFTNL_EXPR_OBJREF_SET_ID:
+               objref->set.id = *((uint32_t *)data);
+               break;
        default:
                return -1;
        }
@@ -59,6 +75,15 @@ static const void *nftnl_expr_objref_get(const struct nftnl_expr *e,
        case NFTNL_EXPR_OBJREF_IMM_NAME:
                *data_len = strlen(objref->imm.name) + 1;
                return objref->imm.name;
+       case NFTNL_EXPR_OBJREF_SET_SREG:
+               *data_len = sizeof(objref->set.sreg);
+               return &objref->set.sreg;
+       case NFTNL_EXPR_OBJREF_SET_NAME:
+               *data_len = strlen(objref->set.name) + 1;
+               return objref->set.name;
+       case NFTNL_EXPR_OBJREF_SET_ID:
+               *data_len = sizeof(objref->set.id);
+               return &objref->set.id;
        }
        return NULL;
 }
@@ -77,9 +102,15 @@ static int nftnl_expr_objref_cb(const struct nlattr *attr, void *data)
                        abi_breakage();
                break;
        case NFTA_OBJREF_IMM_NAME:
+       case NFTA_OBJREF_SET_NAME:
                if (mnl_attr_validate(attr, MNL_TYPE_STRING) < 0)
                        abi_breakage();
                break;
+       case NFTA_OBJREF_SET_SREG:
+       case NFTA_OBJREF_SET_ID:
+               if (mnl_attr_validate(attr, MNL_TYPE_U32) < 0)
+                       abi_breakage();
+               break;
        }
 
        tb[type] = attr;
@@ -96,6 +127,14 @@ static void nftnl_expr_objref_build(struct nlmsghdr *nlh,
                                 htonl(objref->imm.type));
        if (e->flags & (1 << NFTNL_EXPR_OBJREF_IMM_NAME))
                mnl_attr_put_str(nlh, NFTA_OBJREF_IMM_NAME, objref->imm.name);
+       if (e->flags & (1 << NFTNL_EXPR_OBJREF_SET_SREG))
+               mnl_attr_put_u32(nlh, NFTA_OBJREF_SET_SREG,
+                                htonl(objref->set.sreg));
+       if (e->flags & (1 << NFTNL_EXPR_OBJREF_SET_NAME))
+               mnl_attr_put_str(nlh, NFTA_OBJREF_SET_NAME, objref->set.name);
+       if (e->flags & (1 << NFTNL_EXPR_OBJREF_SET_ID))
+               mnl_attr_put_u32(nlh, NFTA_OBJREF_SET_ID,
+                                htonl(objref->set.id));
 }
 
 static int nftnl_expr_objref_parse(struct nftnl_expr *e, struct nlattr *attr)
@@ -116,6 +155,21 @@ static int nftnl_expr_objref_parse(struct nftnl_expr *e, struct nlattr *attr)
                        strdup(mnl_attr_get_str(tb[NFTA_OBJREF_IMM_NAME]));
                e->flags |= (1 << NFTNL_EXPR_OBJREF_IMM_NAME);
        }
+       if (tb[NFTA_OBJREF_SET_SREG]) {
+               objref->set.sreg =
+                       ntohl(mnl_attr_get_u32(tb[NFTA_OBJREF_SET_SREG]));
+               e->flags |= (1 << NFTNL_EXPR_OBJREF_SET_SREG);
+       }
+       if (tb[NFTA_OBJREF_SET_NAME]) {
+               objref->set.name =
+                       strdup(mnl_attr_get_str(tb[NFTA_OBJREF_SET_NAME]));
+               e->flags |= (1 << NFTNL_EXPR_OBJREF_SET_NAME);
+       }
+       if (tb[NFTA_OBJREF_SET_ID]) {
+               objref->set.id =
+                       ntohl(mnl_attr_get_u32(tb[NFTA_OBJREF_SET_ID]));
+               e->flags |= (1 << NFTNL_EXPR_OBJREF_SET_ID);
+       }
 
        return 0;
 }
@@ -157,6 +211,10 @@ static int nftnl_expr_objref_export(char *buf, size_t size,
                nftnl_buf_u32(&b, type, objref->imm.type, BYTES);
        if (e->flags & (1 << NFTNL_EXPR_OBJREF_IMM_NAME))
                nftnl_buf_str(&b, type, objref->imm.name, NAME);
+       if (e->flags & (1 << NFTNL_EXPR_OBJREF_SET_SREG))
+               nftnl_buf_u32(&b, type, objref->set.sreg, SREG);
+       if (e->flags & (1 << NFTNL_EXPR_OBJREF_SET_NAME))
+               nftnl_buf_str(&b, type, objref->set.name, SET);
 
        return nftnl_buf_done(&b);
 }
@@ -166,8 +224,12 @@ static int nftnl_expr_objref_snprintf_default(char *buf, size_t len,
 {
        struct nftnl_expr_objref *objref = nftnl_expr_data(e);
 
-       return snprintf(buf, len, "type %u name %s ",
-                       objref->imm.type, objref->imm.name);
+       if (e->flags & (1 << NFTNL_EXPR_OBJREF_SET_SREG))
+               return snprintf(buf, len, "sreg %u set %s id %u ",
+                               objref->set.sreg, objref->set.name, objref->set.id);
+       else
+               return snprintf(buf, len, "type %u name %s ",
+                               objref->imm.type, objref->imm.name);
 }
 
 static int nftnl_expr_objref_snprintf(char *buf, size_t len, uint32_t type,
@@ -197,6 +259,12 @@ static bool nftnl_expr_objref_cmp(const struct nftnl_expr *e1,
                eq &= (c1->imm.type == c2->imm.type);
        if (e1->flags & (1 << NFTNL_EXPR_OBJREF_IMM_NAME))
                eq &= !strcmp(c1->imm.name, c2->imm.name);
+       if (e1->flags & (1 << NFTNL_EXPR_OBJREF_SET_SREG))
+               eq &= (c1->set.sreg == c2->set.sreg);
+       if (e1->flags & (1 << NFTNL_EXPR_OBJREF_SET_NAME))
+               eq &= !strcmp(c1->set.name, c2->set.name);
+       if (e1->flags & (1 << NFTNL_EXPR_OBJREF_SET_ID))
+               eq &= (c1->set.id == c2->set.id);
 
        return eq;
 }