zoukankan      html  css  js  c++  java
  • Run P4 without P4factory

    /*
    Copyright 2013-present Barefoot Networks, 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.
    */
    
    #include "includes/headers.p4"
    #include "includes/parser.p4"
    #include "includes/intrinsic.p4"
    
    #define FLOWLET_MAP_SIZE 13    // 8K
    #define FLOWLET_INACTIVE_TOUT 50000 // usec -> 50ms
    
    header_type ingress_metadata_t {
        fields {
            flow_ipg : 32; // inter-packet gap
            flowlet_map_index : FLOWLET_MAP_SIZE; // flowlet map index
            flowlet_id : 16; // flowlet id
            flowlet_lasttime : 32; // flowlet's last reference time
    
            ecmp_offset : 14; // offset into the ecmp table
    
            nhop_ipv4 : 32;
        }
    }
    
    metadata ingress_metadata_t ingress_metadata;
    
    action _drop() {
        drop();
    }
    
    field_list l3_hash_fields {
        ipv4.srcAddr;
        ipv4.dstAddr;
        ipv4.protocol;
        tcp.srcPort;
        tcp.dstPort;
    }
    
    field_list_calculation flowlet_map_hash {
        input {
            l3_hash_fields;
        }
        algorithm : crc16;
        output_width : FLOWLET_MAP_SIZE;
    }
    
    register flowlet_lasttime {
        width : 32;
        instance_count : 8192;
    }
    
    register flowlet_id {
        width : 16;
        instance_count : 8192;
    }
    
    action set_nhop(nhop_ipv4, port) {
        modify_field(ingress_metadata.nhop_ipv4, nhop_ipv4);
        modify_field(standard_metadata.egress_spec, port);
        add_to_field(ipv4.ttl, -1);
    }
    
    action lookup_flowlet_map() {
        modify_field_with_hash_based_offset(ingress_metadata.flowlet_map_index, 0,
                                            flowlet_map_hash, FLOWLET_MAP_SIZE);
    
        register_read(ingress_metadata.flowlet_id,
                      flowlet_id, ingress_metadata.flowlet_map_index);
    
        modify_field(ingress_metadata.flow_ipg,
                     intrinsic_metadata.ingress_global_timestamp);
        register_read(ingress_metadata.flowlet_lasttime,
        flowlet_lasttime, ingress_metadata.flowlet_map_index);
        subtract_from_field(ingress_metadata.flow_ipg,
                            ingress_metadata.flowlet_lasttime);
    
        register_write(flowlet_lasttime, ingress_metadata.flowlet_map_index,
                       intrinsic_metadata.ingress_global_timestamp);
    }
    
    table flowlet {
        actions { lookup_flowlet_map; }
    }
    
    action update_flowlet_id() {
        add_to_field(ingress_metadata.flowlet_id, 1);
        register_write(flowlet_id, ingress_metadata.flowlet_map_index,
                       ingress_metadata.flowlet_id);
    }
    
    table new_flowlet {
        actions { update_flowlet_id; }
    }
    
    field_list flowlet_l3_hash_fields {
        ipv4.srcAddr;
        ipv4.dstAddr;
        ipv4.protocol;
        tcp.srcPort;
        tcp.dstPort;
        ingress_metadata.flowlet_id;
    }
    
    #define ECMP_BIT_WIDTH 10
    #define ECMP_GROUP_TABLE_SIZE 1024
    #define ECMP_NHOP_TABLE_SIZE 16384
    
    field_list_calculation flowlet_ecmp_hash {
        input {
            flowlet_l3_hash_fields;
        }
        algorithm : crc16;
        output_width : ECMP_BIT_WIDTH;
    }
    
    action set_ecmp_select(ecmp_base, ecmp_count) {
        modify_field_with_hash_based_offset(ingress_metadata.ecmp_offset, ecmp_base,
                                            flowlet_ecmp_hash, ecmp_count);
    }
    
    table ecmp_group {
        reads {
            ipv4.dstAddr : lpm;
        }
        actions {
            _drop;
            set_ecmp_select;
        }
        size : ECMP_GROUP_TABLE_SIZE;
    }
    
    table ecmp_nhop {
        reads {
            ingress_metadata.ecmp_offset : exact;
        }
        actions {
            _drop;
            set_nhop;
        }
        size : ECMP_NHOP_TABLE_SIZE;
    }
    
    action set_dmac(dmac) {
        modify_field(ethernet.dstAddr, dmac);
    }
    
    table forward {
        reads {
            ingress_metadata.nhop_ipv4 : exact;
        }
        actions {
            set_dmac;
            _drop;
        }
        size: 512;
    }
    
    action rewrite_mac(smac) {
        modify_field(ethernet.srcAddr, smac);
    }
    
    table send_frame {
        reads {
            standard_metadata.egress_port: exact;
        }
        actions {
            rewrite_mac;
            _drop;
        }
        size: 256;
    }
    
    control ingress {
        apply(flowlet);
        if (ingress_metadata.flow_ipg > FLOWLET_INACTIVE_TOUT) {
            apply(new_flowlet);
        }
        apply(ecmp_group);
        apply(ecmp_nhop);
        apply(forward);
    }
    
    control egress {
        apply(send_frame);
    }
    
  • 相关阅读:
    如何删除完全重复的列
    串联多个字符串,引发string和stringBuilder的比较
    借鉴ANJOU的方法改写了上次的TreeView
    如何改变ListBox中内容的顺序
    执行Insert语句时使用string的Format用法
    moss 2007 中FCKEditor编辑器的使用
    qt 获得cmd 命令运行的结果 GIS
    迭代器 GIS
    win32 创建带图片的button GIS
    c++标准库的构成 GIS
  • 原文地址:https://www.cnblogs.com/qq952693358/p/6208799.html
Copyright © 2011-2022 走看看