return ret;
 }
 
-static int bpf_skb_proto_4_to_6(struct sk_buff *skb)
+static int bpf_skb_proto_4_to_6(struct sk_buff *skb, u64 flags)
 {
        const u32 len_diff = sizeof(struct ipv6hdr) - sizeof(struct iphdr);
        u32 off = skb_mac_header_len(skb);
                }
 
                /* Due to IPv6 header, MSS needs to be downgraded. */
-               skb_decrease_gso_size(shinfo, len_diff);
+               if (!(flags & BPF_F_ADJ_ROOM_FIXED_GSO))
+                       skb_decrease_gso_size(shinfo, len_diff);
+
                /* Header must be checked, and gso_segs recomputed. */
                shinfo->gso_type |= SKB_GSO_DODGY;
                shinfo->gso_segs = 0;
        return 0;
 }
 
-static int bpf_skb_proto_6_to_4(struct sk_buff *skb)
+static int bpf_skb_proto_6_to_4(struct sk_buff *skb, u64 flags)
 {
        const u32 len_diff = sizeof(struct ipv6hdr) - sizeof(struct iphdr);
        u32 off = skb_mac_header_len(skb);
                }
 
                /* Due to IPv4 header, MSS can be upgraded. */
-               skb_increase_gso_size(shinfo, len_diff);
+               if (!(flags & BPF_F_ADJ_ROOM_FIXED_GSO))
+                       skb_increase_gso_size(shinfo, len_diff);
+
                /* Header must be checked, and gso_segs recomputed. */
                shinfo->gso_type |= SKB_GSO_DODGY;
                shinfo->gso_segs = 0;
        return 0;
 }
 
-static int bpf_skb_proto_xlat(struct sk_buff *skb, __be16 to_proto)
+static int bpf_skb_proto_xlat(struct sk_buff *skb, __be16 to_proto, u64 flags)
 {
        __be16 from_proto = skb->protocol;
 
        if (from_proto == htons(ETH_P_IP) &&
              to_proto == htons(ETH_P_IPV6))
-               return bpf_skb_proto_4_to_6(skb);
+               return bpf_skb_proto_4_to_6(skb, flags);
 
        if (from_proto == htons(ETH_P_IPV6) &&
              to_proto == htons(ETH_P_IP))
-               return bpf_skb_proto_6_to_4(skb);
+               return bpf_skb_proto_6_to_4(skb, flags);
 
        return -ENOTSUPP;
 }
 {
        int ret;
 
-       if (unlikely(flags))
+       if (unlikely(flags & ~(BPF_F_ADJ_ROOM_FIXED_GSO)))
                return -EINVAL;
 
        /* General idea is that this helper does the basic groundwork
         * that. For offloads, we mark packet as dodgy, so that headers
         * need to be verified first.
         */
-       ret = bpf_skb_proto_xlat(skb, proto);
+       ret = bpf_skb_proto_xlat(skb, proto, flags);
        bpf_compute_data_pointers(skb);
        return ret;
 }