From: Ethan Jackson Date: Fri, 4 Feb 2011 02:33:53 +0000 (-0800) Subject: vswitchd: Process special packets more aggressively. X-Git-Url: https://pintos-os.org/cgi-bin/gitweb.cgi?a=commitdiff_plain;h=ebe482fd2119ef322751603f46e033bb374a8f11;p=openvswitch vswitchd: Process special packets more aggressively. Before this patch, special packets such as LACP and CFM messages were only processed if they had NORMAL open flow actions. With this patch these messages are always processed unless originated in ofproto_send_packet(). --- diff --git a/ofproto/ofproto.c b/ofproto/ofproto.c index e0f1b663..39d34573 100644 --- a/ofproto/ofproto.c +++ b/ofproto/ofproto.c @@ -125,6 +125,11 @@ struct action_xlate_ctx { * calling action_xlate_ctx_init(). */ void (*resubmit_hook)(struct action_xlate_ctx *, const struct rule *); + /* If true, the speciality of 'flow' should be checked before executing + * its actions. If special_cb returns false on 'flow' rendered + * uninstallable and no actions will be executed. */ + bool check_special; + /* xlate_actions() initializes and uses these members. The client might want * to look at them after it returns. */ @@ -1446,6 +1451,8 @@ ofproto_send_packet(struct ofproto *p, const struct flow *flow, struct ofpbuf *odp_actions; action_xlate_ctx_init(&ctx, p, flow, packet); + /* Always xlate packets originated in this function. */ + ctx.check_special = false; odp_actions = xlate_actions(&ctx, actions, n_actions); /* XXX Should we translate the dpif_execute() errno value into an OpenFlow @@ -3110,6 +3117,7 @@ action_xlate_ctx_init(struct action_xlate_ctx *ctx, ctx->flow = *flow; ctx->packet = packet; ctx->resubmit_hook = NULL; + ctx->check_special = true; } static struct ofpbuf * @@ -3124,7 +3132,16 @@ xlate_actions(struct action_xlate_ctx *ctx, ctx->nf_output_iface = NF_OUT_DROP; ctx->recurse = 0; ctx->last_pop_priority = -1; - do_xlate_actions(in, n_in, ctx); + + if (!ctx->check_special + || (ctx->ofproto->ofhooks->special_cb + && ctx->ofproto->ofhooks->special_cb(&ctx->flow, ctx->packet, + ctx->ofproto->aux))) { + do_xlate_actions(in, n_in, ctx); + } else { + ctx->may_set_up_flow = false; + } + remove_pop_action(ctx); /* Check with in-band control to see if we're allowed to set up this @@ -4370,6 +4387,12 @@ handle_miss_upcall(struct ofproto *p, struct dpif_upcall *upcall) /* Set header pointers in 'flow'. */ flow_extract(upcall->packet, flow.tun_id, flow.in_port, &flow); + if (p->ofhooks->special_cb + && !p->ofhooks->special_cb(&flow, upcall->packet, p->aux)) { + ofpbuf_delete(upcall->packet); + return; + } + /* Check with in-band control to see if this packet should be sent * to the local port regardless of the flow table. */ if (in_band_msg_in_hook(p->in_band, &flow, upcall->packet)) { @@ -5126,5 +5149,6 @@ default_normal_ofhook_cb(const struct flow *flow, const struct ofpbuf *packet, static const struct ofhooks default_ofhooks = { default_normal_ofhook_cb, NULL, + NULL, NULL }; diff --git a/ofproto/ofproto.h b/ofproto/ofproto.h index 8e4e2a65..7516068f 100644 --- a/ofproto/ofproto.h +++ b/ofproto/ofproto.h @@ -149,6 +149,8 @@ struct ofhooks { bool (*normal_cb)(const struct flow *, const struct ofpbuf *packet, struct ofpbuf *odp_actions, tag_type *, uint16_t *nf_output_iface, void *aux); + bool (*special_cb)(const struct flow *flow, const struct ofpbuf *packet, + void *aux); void (*account_flow_cb)(const struct flow *, tag_type tags, const struct nlattr *odp_actions, size_t actions_len, diff --git a/vswitchd/bridge.c b/vswitchd/bridge.c index 72eb7346..f5c5af37 100644 --- a/vswitchd/bridge.c +++ b/vswitchd/bridge.c @@ -76,6 +76,8 @@ VLOG_DEFINE_THIS_MODULE(bridge); COVERAGE_DEFINE(bridge_flush); COVERAGE_DEFINE(bridge_process_flow); +COVERAGE_DEFINE(bridge_process_cfm); +COVERAGE_DEFINE(bridge_process_lacp); COVERAGE_DEFINE(bridge_reconfigure); COVERAGE_DEFINE(bridge_lacp_update); @@ -3005,26 +3007,38 @@ bridge_normal_ofhook_cb(const struct flow *flow, const struct ofpbuf *packet, struct ofpbuf *actions, tag_type *tags, uint16_t *nf_output_iface, void *br_) { - struct iface *iface; struct bridge *br = br_; COVERAGE_INC(bridge_process_flow); + return process_flow(br, flow, packet, actions, tags, nf_output_iface); +} + +static bool +bridge_special_ofhook_cb(const struct flow *flow, + const struct ofpbuf *packet, void *br_) +{ + struct iface *iface; + struct bridge *br = br_; iface = iface_from_dp_ifidx(br, flow->in_port); if (cfm_should_process_flow(flow)) { - if (packet && iface->cfm) { + + if (iface && packet && iface->cfm) { + COVERAGE_INC(bridge_process_cfm); cfm_process_heartbeat(iface->cfm, packet); } return false; } else if (flow->dl_type == htons(ETH_TYPE_LACP)) { - if (packet) { + + if (iface && packet) { + COVERAGE_INC(bridge_process_lacp); lacp_process_packet(packet, iface); } return false; } - return process_flow(br, flow, packet, actions, tags, nf_output_iface); + return true; } static void @@ -3094,6 +3108,7 @@ bridge_account_checkpoint_ofhook_cb(void *br_) static struct ofhooks bridge_ofhooks = { bridge_normal_ofhook_cb, + bridge_special_ofhook_cb, bridge_account_flow_ofhook_cb, bridge_account_checkpoint_ofhook_cb, };