diff options
author | Ben Pfaff <blp@nicira.com> | 2010-11-10 14:39:54 -0800 |
---|---|---|
committer | Ben Pfaff <blp@nicira.com> | 2010-11-22 10:11:40 -0800 |
commit | d8ae4d672673cd72285eb405a96b4ac3590a7639 (patch) | |
tree | fb6c030b086cc5f2b492d3c4411d12d81c9093b3 /tests/test-classifier.c | |
parent | 844dff325b1f6a6f520fce9242c85162275ab7ad (diff) | |
download | openvswitch-d8ae4d672673cd72285eb405a96b4ac3590a7639.tar.gz |
flow: Fully separate flow_wildcards from OpenFlow wildcard bits.
Originally, wildcards were just the OpenFlow OFPFW_* bits. Then, when
OpenFlow added CIDR masks for IP addresses, struct flow_wildcards was born
with additional members for those masks, derived from the wildcard bits.
Then, when OVS added support for tunnels, we added another bit
NXFW_TUN_ID that coexisted with the OFPFW_*. Later we added even more bits
that do not appear in the OpenFlow 1.0 match structure at all. This had
become really confusing, and the difficulties were especially visible in
the long list of invariants in comments on struct flow_wildcards.
This commit cleanly separates the OpenFlow 1.0 wildcard bits from the
bits used inside Open vSwitch, by defining a new set of bits that are
used only internally to Open vSwitch and converting to and from those
wildcard bits at the point where data comes off or goes onto the wire.
It also moves those functions into ofp-util.[ch] since they are only for
dealing with OpenFlow wire protocol now.
Diffstat (limited to 'tests/test-classifier.c')
-rw-r--r-- | tests/test-classifier.c | 100 |
1 files changed, 46 insertions, 54 deletions
diff --git a/tests/test-classifier.c b/tests/test-classifier.c index 07bd0294f..e8434f0a7 100644 --- a/tests/test-classifier.c +++ b/tests/test-classifier.c @@ -40,24 +40,23 @@ #include <assert.h> /* Fields in a rule. */ -#define CLS_FIELDS \ - /* struct flow all-caps */ \ - /* wildcard bit(s) member name name */ \ - /* ----------------- ----------- -------- */ \ - CLS_FIELD(NXFW_TUN_ID, tun_id, TUN_ID) \ - CLS_FIELD(OFPFW_NW_SRC_MASK, nw_src, NW_SRC) \ - CLS_FIELD(OFPFW_NW_DST_MASK, nw_dst, NW_DST) \ - CLS_FIELD(OFPFW_IN_PORT, in_port, IN_PORT) \ - CLS_FIELD(OFPFW_DL_VLAN, dl_vlan, DL_VLAN) \ - CLS_FIELD(OFPFW_DL_TYPE, dl_type, DL_TYPE) \ - CLS_FIELD(OFPFW_TP_SRC, tp_src, TP_SRC) \ - CLS_FIELD(OFPFW_TP_DST, tp_dst, TP_DST) \ - CLS_FIELD(OFPFW_DL_SRC, dl_src, DL_SRC) \ - CLS_FIELD(OFPFW_DL_DST | FWW_ETH_MCAST, \ - dl_dst, DL_DST) \ - CLS_FIELD(OFPFW_NW_PROTO, nw_proto, NW_PROTO) \ - CLS_FIELD(OFPFW_DL_VLAN_PCP, dl_vlan_pcp, DL_VLAN_PCP) \ - CLS_FIELD(OFPFW_NW_TOS, nw_tos, NW_TOS) +#define CLS_FIELDS \ + /* struct flow all-caps */ \ + /* FWW_* bit(s) member name name */ \ + /* -------------------------- ----------- -------- */ \ + CLS_FIELD(FWW_TUN_ID, tun_id, TUN_ID) \ + CLS_FIELD(0, nw_src, NW_SRC) \ + CLS_FIELD(0, nw_dst, NW_DST) \ + CLS_FIELD(FWW_IN_PORT, in_port, IN_PORT) \ + CLS_FIELD(FWW_DL_VLAN, dl_vlan, DL_VLAN) \ + CLS_FIELD(FWW_DL_TYPE, dl_type, DL_TYPE) \ + CLS_FIELD(FWW_TP_SRC, tp_src, TP_SRC) \ + CLS_FIELD(FWW_TP_DST, tp_dst, TP_DST) \ + CLS_FIELD(FWW_DL_SRC, dl_src, DL_SRC) \ + CLS_FIELD(FWW_DL_DST | FWW_ETH_MCAST, dl_dst, DL_DST) \ + CLS_FIELD(FWW_NW_PROTO, nw_proto, NW_PROTO) \ + CLS_FIELD(FWW_DL_VLAN_PCP, dl_vlan_pcp, DL_VLAN_PCP) \ + CLS_FIELD(FWW_NW_TOS, nw_tos, NW_TOS) /* Field indexes. * @@ -73,7 +72,7 @@ enum { struct cls_field { int ofs; /* Offset in struct flow. */ int len; /* Length in bytes. */ - uint32_t wildcards; /* OFPFW_* bit or bits for this field. */ + flow_wildcards_t wildcards; /* FWW_* bit or bits for this field. */ const char *name; /* Name (for debugging). */ }; @@ -189,30 +188,24 @@ match(const struct cls_rule *wild, const struct flow *fixed) for (f_idx = 0; f_idx < CLS_N_FIELDS; f_idx++) { const struct cls_field *f = &cls_fields[f_idx]; - void *wild_field = (char *) &wild->flow + f->ofs; - void *fixed_field = (char *) fixed + f->ofs; - - if ((wild->wc.wildcards & f->wildcards) == f->wildcards || - !memcmp(wild_field, fixed_field, f->len)) { - /* Definite match. */ - continue; + bool eq; + + if (f->wildcards) { + void *wild_field = (char *) &wild->flow + f->ofs; + void *fixed_field = (char *) fixed + f->ofs; + eq = ((wild->wc.wildcards & f->wildcards) == f->wildcards + || !memcmp(wild_field, fixed_field, f->len)); + } else if (f_idx == CLS_F_IDX_NW_SRC) { + eq = !((fixed->nw_src ^ wild->flow.nw_src) & wild->wc.nw_src_mask); + } else if (f_idx == CLS_F_IDX_NW_DST) { + eq = !((fixed->nw_dst ^ wild->flow.nw_dst) & wild->wc.nw_dst_mask); + } else { + NOT_REACHED(); } - if (wild->wc.wildcards & f->wildcards) { - uint32_t test = get_unaligned_u32(wild_field); - uint32_t ip = get_unaligned_u32(fixed_field); - uint32_t mask; - int shift; - - shift = (f_idx == CLS_F_IDX_NW_SRC - ? OFPFW_NW_SRC_SHIFT : OFPFW_NW_DST_SHIFT); - mask = ofputil_wcbits_to_netmask(wild->wc.wildcards >> shift); - if (!((test ^ ip) & mask)) { - continue; - } + if (!eq) { + return false; } - - return false; } return true; } @@ -456,27 +449,26 @@ static struct test_rule * make_rule(int wc_fields, unsigned int priority, int value_pat) { const struct cls_field *f; - struct flow_wildcards wc; struct test_rule *rule; - uint32_t wildcards; - struct flow flow; - wildcards = 0; - memset(&flow, 0, sizeof flow); + rule = xzalloc(sizeof *rule); + cls_rule_init_catchall(&rule->cls_rule, wc_fields ? priority : UINT_MAX); for (f = &cls_fields[0]; f < &cls_fields[CLS_N_FIELDS]; f++) { int f_idx = f - cls_fields; - if (wc_fields & (1u << f_idx)) { - wildcards |= f->wildcards; + int value_idx = (value_pat & (1u << f_idx)) != 0; + memcpy((char *) &rule->cls_rule.flow + f->ofs, + values[f_idx][value_idx], f->len); + + if (f->wildcards) { + rule->cls_rule.wc.wildcards &= ~f->wildcards; + } else if (f_idx == CLS_F_IDX_NW_SRC) { + rule->cls_rule.wc.nw_src_mask = htonl(UINT32_MAX); + } else if (f_idx == CLS_F_IDX_NW_DST) { + rule->cls_rule.wc.nw_dst_mask = htonl(UINT32_MAX); } else { - int value_idx = (value_pat & (1u << f_idx)) != 0; - memcpy((char *) &flow + f->ofs, values[f_idx][value_idx], f->len); + NOT_REACHED(); } } - - rule = xzalloc(sizeof *rule); - flow_wildcards_init(&wc, wildcards); - cls_rule_init(&flow, &wc, !wildcards ? UINT_MAX : priority, - &rule->cls_rule); return rule; } |