| /* |
| * QEMU TX packets abstraction |
| * |
| * Copyright (c) 2012 Ravello Systems LTD (http://ravellosystems.com) |
| * |
| * Developed by Daynix Computing LTD (http://www.daynix.com) |
| * |
| * Authors: |
| * Dmitry Fleytman <dmitry@daynix.com> |
| * Tamir Shomer <tamirs@daynix.com> |
| * Yan Vugenfirer <yan@daynix.com> |
| * |
| * This work is licensed under the terms of the GNU GPL, version 2 or later. |
| * See the COPYING file in the top-level directory. |
| * |
| */ |
| |
| #ifndef NET_TX_PKT_H |
| #define NET_TX_PKT_H |
| |
| #include "net/eth.h" |
| #include "exec/hwaddr.h" |
| |
| /* define to enable packet dump functions */ |
| /*#define NET_TX_PKT_DEBUG*/ |
| |
| struct NetTxPkt; |
| |
| typedef void (*NetTxPktFreeFrag)(void *, void *, size_t); |
| typedef void (*NetTxPktSend)(void *, const struct iovec *, int, const struct iovec *, int); |
| |
| /** |
| * Init function for tx packet functionality |
| * |
| * @pkt: packet pointer |
| * @max_frags: max tx ip fragments |
| */ |
| void net_tx_pkt_init(struct NetTxPkt **pkt, uint32_t max_frags); |
| |
| /** |
| * Clean all tx packet resources. |
| * |
| * @pkt: packet. |
| */ |
| void net_tx_pkt_uninit(struct NetTxPkt *pkt); |
| |
| /** |
| * get virtio header |
| * |
| * @pkt: packet |
| * @ret: virtio header |
| */ |
| struct virtio_net_hdr *net_tx_pkt_get_vhdr(struct NetTxPkt *pkt); |
| |
| /** |
| * build virtio header (will be stored in module context) |
| * |
| * @pkt: packet |
| * @tso_enable: TSO enabled |
| * @csum_enable: CSO enabled |
| * @gso_size: MSS size for TSO |
| * @ret: operation result |
| * |
| */ |
| bool net_tx_pkt_build_vheader(struct NetTxPkt *pkt, bool tso_enable, |
| bool csum_enable, uint32_t gso_size); |
| |
| /** |
| * updates vlan tag, and adds vlan header with custom ethernet type |
| * in case it is missing. |
| * |
| * @pkt: packet |
| * @vlan: VLAN tag |
| * @vlan_ethtype: VLAN header Ethernet type |
| * |
| */ |
| void net_tx_pkt_setup_vlan_header_ex(struct NetTxPkt *pkt, |
| uint16_t vlan, uint16_t vlan_ethtype); |
| |
| /** |
| * updates vlan tag, and adds vlan header in case it is missing |
| * |
| * @pkt: packet |
| * @vlan: VLAN tag |
| * |
| */ |
| static inline void |
| net_tx_pkt_setup_vlan_header(struct NetTxPkt *pkt, uint16_t vlan) |
| { |
| net_tx_pkt_setup_vlan_header_ex(pkt, vlan, ETH_P_VLAN); |
| } |
| |
| /** |
| * populate data fragment into pkt context. |
| * |
| * @pkt: packet |
| * @base: pointer to fragment |
| * @len: length of fragment |
| * |
| */ |
| bool net_tx_pkt_add_raw_fragment(struct NetTxPkt *pkt, void *base, size_t len); |
| |
| /** |
| * Fix ip header fields and calculate IP header and pseudo header checksums. |
| * |
| * @pkt: packet |
| * |
| */ |
| void net_tx_pkt_update_ip_checksums(struct NetTxPkt *pkt); |
| |
| /** |
| * Calculate the IP header checksum. |
| * |
| * @pkt: packet |
| * |
| */ |
| void net_tx_pkt_update_ip_hdr_checksum(struct NetTxPkt *pkt); |
| |
| /** |
| * Calculate the SCTP checksum. |
| * |
| * @pkt: packet |
| * |
| */ |
| bool net_tx_pkt_update_sctp_checksum(struct NetTxPkt *pkt); |
| |
| /** |
| * get length of all populated data. |
| * |
| * @pkt: packet |
| * @ret: total data length |
| * |
| */ |
| size_t net_tx_pkt_get_total_len(struct NetTxPkt *pkt); |
| |
| /** |
| * get packet type |
| * |
| * @pkt: packet |
| * @ret: packet type |
| * |
| */ |
| eth_pkt_types_e net_tx_pkt_get_packet_type(struct NetTxPkt *pkt); |
| |
| /** |
| * prints packet data if debug is enabled |
| * |
| * @pkt: packet |
| * |
| */ |
| void net_tx_pkt_dump(struct NetTxPkt *pkt); |
| |
| /** |
| * reset tx packet private context (needed to be called between packets) |
| * |
| * @pkt: packet |
| * @callback: function to free the fragments |
| * @context: pointer to be passed to the callback |
| */ |
| void net_tx_pkt_reset(struct NetTxPkt *pkt, |
| NetTxPktFreeFrag callback, void *context); |
| |
| /** |
| * Unmap a fragment mapped from a PCI device. |
| * |
| * @context: PCI device owning fragment |
| * @base: pointer to fragment |
| * @len: length of fragment |
| */ |
| void net_tx_pkt_unmap_frag_pci(void *context, void *base, size_t len); |
| |
| /** |
| * map data fragment from PCI device and populate it into pkt context. |
| * |
| * @pci_dev: PCI device owning fragment |
| * @pa: physical address of fragment |
| * @len: length of fragment |
| */ |
| bool net_tx_pkt_add_raw_fragment_pci(struct NetTxPkt *pkt, PCIDevice *pci_dev, |
| dma_addr_t pa, size_t len); |
| |
| /** |
| * Send packet to qemu. handles sw offloads if vhdr is not supported. |
| * |
| * @pkt: packet |
| * @nc: NetClientState |
| * @ret: operation result |
| * |
| */ |
| bool net_tx_pkt_send(struct NetTxPkt *pkt, NetClientState *nc); |
| |
| /** |
| * Send packet with a custom function. |
| * |
| * @pkt: packet |
| * @offload: whether the callback implements offloading |
| * @callback: a function to be called back for each transformed packet |
| * @context: a pointer to be passed to the callback. |
| * @ret: operation result |
| */ |
| bool net_tx_pkt_send_custom(struct NetTxPkt *pkt, bool offload, |
| NetTxPktSend callback, void *context); |
| |
| /** |
| * parse raw packet data and analyze offload requirements. |
| * |
| * @pkt: packet |
| * |
| */ |
| bool net_tx_pkt_parse(struct NetTxPkt *pkt); |
| |
| /** |
| * indicates if there are data fragments held by this packet object. |
| * |
| * @pkt: packet |
| * |
| */ |
| bool net_tx_pkt_has_fragments(struct NetTxPkt *pkt); |
| |
| /** |
| * Fix IPv6 'plen' field. |
| * If ipv6 payload length field is 0 - then there should be Hop-by-Hop |
| * option for packets greater than 65,535. |
| * For packets with a payload less than 65,535: fix 'plen' field. |
| * For backends with vheader, we need just one packet with proper |
| * payload size. For now, qemu drops every packet with size greater 64K |
| * (see net_tx_pkt_send()) so, there is no reason to add jumbo option to ip6 |
| * hop-by-hop extension if it's missed |
| * |
| * @pkt packet |
| */ |
| void net_tx_pkt_fix_ip6_payload_len(struct NetTxPkt *pkt); |
| |
| #endif |