summaryrefslogtreecommitdiff
path: root/datapath-windows/ovsext/Ip6Fragment.h
blob: f978bf5c3f113028cef0ff76aaa9d733e9a1db9a (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
/*
 * Copyright (c) 2022 VMware, Inc.
 *
 * 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.
 */

#ifndef __IP6FRAGMENT_H_
#define __IP6FRAGMENT_H_ 1
#include "PacketIO.h"

typedef struct _OVS_FRAGMENT6_LIST {
    CHAR *pbuff;
    UINT16 len; /* Fragment data length. */
    UINT16 offset; /* Fragment data offset. */
    struct _OVS_FRAGMENT6_LIST *next;
} OVS_FRAGMENT6_LIST, *POVS_FRAGMENT6_LIST;

typedef struct _OVS_IP6FRAG_KEY {
    UINT8 protocol;
    UINT8 pad_1[3];             /* Align the structure to address boundaries.*/
    UINT32 id;
    struct in6_addr sAddr;
    struct in6_addr dAddr;
    ovs_be64 tunnelId;
} OVS_IP6FRAG_KEY, *POVS_IP6FRAG_KEY;

typedef struct _OVS_IP6FRAG_ENTRY {
    NDIS_SPIN_LOCK lockObj;       /* To access the entry. */
    BOOLEAN markedForDelete;
    UINT8 numFragments;
    UINT16 totalLen; /* The packet data total length(not
                      * include ipv6 header and opt header length) before
                      * fragment */
    UINT16 recvdLen; /* Total data length packet contains has received */
    UINT16 mru; /* Max receive unit(it's the whole ethernet frame
                 * packet length), it will be used in sent out before forward */
    UINT64 expiration;
    /* refer https://www.rfc-editor.org/rfc/rfc8200.html */
    PCHAR beforeFragHdrBuf;/* ipv6 extension header buf before fragment field */
    UINT16 beforeFragHdrLen;
    UINT16 priorFragEleOffset;/* The last element before fragment field offset */
    PCHAR fragHdrBuf;
    UINT16 fragHdrLen;
    PCHAR behindFragHdrBuf;/* ipv6 extension header buf behind fragment field */
    UINT16 behindFragHdrLen;
    OVS_IP6FRAG_KEY fragKey;
    POVS_FRAGMENT6_LIST head;
    POVS_FRAGMENT6_LIST tail;
    LIST_ENTRY link;
} OVS_IP6FRAG_ENTRY, *POVS_IP6FRAG_ENTRY;

typedef struct _IP6_PktExtHeader_Meta {
    UINT8 firstHdr;
    UINT8 protocol;
    UINT16 beforeFragExtHdrLen;
    UINT16 fragExtHdrLen;
    UINT16 behindFragExtHdrLen;
    UINT16 extHdrTotalLen;
    UINT16 dataPayloadLen;/* Ipv6 data length, not include extension header */
    UINT16 fragOffset;
    UINT16 priorFragEleOffset;
    UINT16 flags;
    UINT16 pktMru;
    UINT32 ident;
    PCHAR beforeFragElePtr;
    IPv6ExtHdr *firstHdrPtr;
} IP6_PktExtHeader_Meta, *PIP6_PktExtHeader_Meta;

typedef struct _OVS_IP6FRAG_THREAD_CTX {
    KEVENT event;
    PVOID threadObject;
    UINT32 exit;
} OVS_IP6FRAG_THREAD_CTX, *POVS_IP6FRAG_THREAD_CTX;

#define IP6_FRAG_HASH_TABLE_SIZE ((UINT32)1 << 10)
#define IP6_FRAG_HASH_TABLE_MASK (IP6_FRAG_HASH_TABLE_SIZE - 1)

#define IP6FRAG_ENTRY_TIMEOUT 300000000LL
#define IP6FRAG_CLEANUP_INTERVAL IP6FRAG_ENTRY_TIMEOUT * 2 /*1m.*/

NDIS_STATUS OvsProcessIpv6Fragment(POVS_SWITCH_CONTEXT switchContext,
                       PNET_BUFFER_LIST *curNbl,
                       OvsCompletionList *completionList,
                       NDIS_SWITCH_PORT_ID sourcePort,
                       POVS_PACKET_HDR_INFO layers,
                       ovs_be64 tunnelId, OvsFlowKey *key);
NDIS_STATUS OvsStorageIpv6ExtHeader(POVS_IP6FRAG_ENTRY entry,
                                    UINT16 beforeFragHdrLen,
                                    UINT16 fragHdrLen,
                                    UINT16 behindFragHdrLen,
                                    UINT16 priorFragEleOffset,
                                    CHAR *pktBuf,
                                    POVS_PACKET_HDR_INFO layers);
NDIS_STATUS OvsInitIp6Fragment(POVS_SWITCH_CONTEXT context);
VOID OvsCleanupIp6Fragment(VOID);
NDIS_STATUS OvsGetPacketMeta(PIP6_PktExtHeader_Meta pktMeta, EthHdr *eth,
                             OvsFlowKey *key, POVS_PACKET_HDR_INFO layers);
PCHAR OvsBuildNewIpv6Hdr(EthHdr *eth, POVS_IP6FRAG_ENTRY entry,
                         POVS_PACKET_HDR_INFO layers, UINT32 *pktLen);

#endif //_IP6FRAGMENT_H_