/* // Copyright (c) 2010-2017 Intel Corporation // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. */ #include "genl4_stream_udp.h" #include "mbuf_utils.h" int stream_udp_is_ended(struct stream_ctx *ctx) { return ctx->cur_action == ctx->stream_cfg->n_actions; } static void update_token_times(struct stream_ctx *ctx) { uint64_t now = rte_rdtsc(); token_time_update(&ctx->token_time_other, now); token_time_update(&ctx->token_time, now); } int stream_udp_proc(struct stream_ctx *ctx, struct rte_mbuf *mbuf, struct l4_meta *l4_meta, uint64_t *next_tsc) { update_token_times(ctx); if (l4_meta) { enum l4gen_peer peer = ctx->stream_cfg->actions[ctx->cur_action].peer; plogx_dbg("Consuming UDP data\n"); /* data should come from the other side */ if (peer == ctx->peer) { plogx_err("Wrong peer\n"); return -1; } /* Fixed length data expected */ if (ctx->stream_cfg->actions[ctx->cur_action].len != l4_meta->len) { plogx_dbg("unexpected UDP len (expected = %u, got = %u, action = %u)\n", ctx->stream_cfg->actions[ctx->cur_action].len, l4_meta->len, ctx->cur_action); return -1; } /* With specific payload */ if (memcmp(ctx->stream_cfg->data[peer].content + ctx->stream_cfg->actions[ctx->cur_action].beg, l4_meta->payload, l4_meta->len) != 0) { plogx_dbg("Bad payload at action_id %d, with peer = %d and pos = %d and len=%d\n", ctx->cur_action, peer, ctx->cur_pos[peer], l4_meta->len); return -1; } ctx->cur_pos[peer] += l4_meta->len; ctx->cur_action++; if (stream_udp_is_ended(ctx)) return -1; token_time_take(&ctx->token_time_other, mbuf_wire_size(mbuf)); /* Time before next packet is expected to arrive. Note, addition amount of time is accounted for due to rate limiting. */ uint64_t wait = token_time_tsc_until_full(&ctx->token_time_other); *next_tsc = wait + ctx->stream_cfg->tsc_timeout; } if (ctx->stream_cfg->actions[ctx->cur_action].peer != ctx->peer) { const char *other_peer_str = ctx->peer != PEER_SERVER? "server" : "client"; plogx_dbg("Expecting more UDP data from %s, will expire = %s\n", other_peer_str, l4_meta == NULL? "yes" : "no"); if (!l4_meta) { ctx->flags |= STREAM_CTX_F_EXPIRED; } return -1; } uint64_t wait_tsc = token_time_tsc_until_full(&ctx->token_time); if (wait_tsc != 0) { plogx_dbg("Wait = %"PRIu64"\n", wait_tsc); *next_tsc = wait_tsc; return -1; } const struct stream_cfg *stream_cfg = ctx->stream_cfg; uint8_t *pkt = rte_pktmbuf_mtod(mbuf, uint8_t *); const struct peer_action *act = &stream_cfg->actions[ctx->cur_action]; uint16_t pkt_len = stream_cfg->data[act->peer].hdr_len + sizeof(struct udp_hdr) + act->len; rte_pktmbuf_pkt_len(mbuf) = pkt_len; rte_pktmbuf_data_len(mbuf) = pkt_len; plogx_dbg("Creating UDP data (peer = %s, payload len = %u)\n", act->peer == PEER_CLIENT? "client" : "server", act->len); /* Construct the packet. The template is used up to L4 header, a gap of sizeof(l4_hdr) is skipped, followed by the payload. */ rte_memcpy(pkt, stream_cfg->data[act->peer].hdr, stream_cfg->data[act->peer].hdr_len); rte_memcpy(pkt + stream_cfg->data[act->peer].hdr_len + sizeof(struct udp_hdr), stream_cfg->data[act->peer].content + act->beg, act->len); struct ipv4_hdr *l3_hdr = (struct ipv4_hdr*)&pkt[stream_cfg->data[act->peer].hdr_len - sizeof(struct ipv4_hdr)]; struct udp_hdr *l4_hdr = (struct udp_hdr*)&pkt[stream_cfg->data[act->peer].hdr_len]; l3_hdr->src_addr = ctx->tuple->dst_addr; l3_hdr->dst_addr = ctx->tuple->src_addr; l3_hdr->next_proto_id = IPPROTO_UDP; l4_hdr->src_port = ctx->tuple->dst_port; l4_hdr->dst_port = ctx->tuple->src_port; l4_hdr->dgram_len = rte_bswap16(sizeof(struct udp_hdr) + act->len); /* TODO: UDP checksum calculation */ l3_hdr->total_length = rte_bswap16(sizeof(struct ipv4_hdr) + sizeof(struct udp_hdr) + act->len); ctx->cur_pos[ctx->peer] += act->len; ctx->cur_action++; /* When the stream has ended, there is no need to schedule another timeout (which will be unscheduled at the end of the stream). */ if (stream_udp_is_ended(ctx)) return 0; token_time_take(&ctx->token_time, mbuf_wire_size(mbuf)); /* Send next packet as soon as possible */ if (ctx->stream_cfg->actions[ctx->cur_action].peer == ctx->peer) { *next_tsc = token_time_tsc_until_full(&ctx->token_time); } else { uint64_t wait = token_time_tsc_until_full(&ctx->token_time_other); *next_tsc = wait + ctx->stream_cfg->tsc_timeout; } return 0; } uint16_t stream_udp_reply_len(struct stream_ctx *ctx) { if (stream_udp_is_ended(ctx)) return 0; else if (ctx->stream_cfg->actions[ctx->cur_action].peer == ctx->peer) return 0; else return ctx->stream_cfg->data[ctx->stream_cfg->actions[ctx->cur_action].peer].hdr_len + sizeof(struct udp_hdr) + ctx->stream_cfg->actions[ctx->cur_action].len; } void stream_udp_calc_len(struct stream_cfg *cfg, uint32_t *n_pkts, uint32_t *n_bytes) { const uint32_t client_hdr_len = cfg->data[PEER_CLIENT].hdr_len; const uint32_t server_hdr_len = cfg->data[PEER_SERVER].hdr_len; *n_pkts = 0; *n_bytes = 0; for (uint32_t i = 0; i < cfg->n_actions; ++i) { const uint32_t send_hdr_len = cfg->actions[i].peer == PEER_CLIENT? client_hdr_len : server_hdr_len; uint32_t len = send_hdr_len + sizeof(struct udp_hdr) + cfg->actions[i].len; *n_bytes += (len < 60? 60 : len) + 24; (*n_pkts)++; } }