diff options
author | Vadim Kochan <vadim4j@gmail.com> | 2016-09-22 23:55:36 +0300 |
---|---|---|
committer | Tobias Klauser <tklauser@distanz.ch> | 2016-10-04 17:19:04 +0200 |
commit | 8aa7e5bf95e452cec43412036b59c04014149cde (patch) | |
tree | c4868f49e2455f3839656686eff01ef7242d600c /trafgen_l2.c | |
parent | d9ae1dc5ab0c71174a5ef56a770fb12a51911839 (diff) |
trafgen: l2: Add support for PFC header
Add PFC header fields setting (prio vector, time per priority).
By default Ethernet header is pushed with required fields.
Signed-off-by: Vadim Kochan <vadim4j@gmail.com>
Signed-off-by: Tobias Klauser <tklauser@distanz.ch>
Diffstat (limited to 'trafgen_l2.c')
-rw-r--r-- | trafgen_l2.c | 42 |
1 files changed, 42 insertions, 0 deletions
diff --git a/trafgen_l2.c b/trafgen_l2.c index 8d2d285..b27c9dd 100644 --- a/trafgen_l2.c +++ b/trafgen_l2.c @@ -31,6 +31,7 @@ static uint16_t pid_to_eth(enum proto_id pid) case PROTO_VLAN: return ETH_P_8021Q; case PROTO_PAUSE: + case PROTO_PFC: return ETH_P_PAUSE; default: bug(); @@ -80,6 +81,46 @@ static struct proto_ops pause_proto_ops = { .header_init = pause_header_init, }; +static struct proto_field pfc_fields[] = { + { .id = PFC_OPCODE, .len = 2, .offset = 0 }, + { .id = PFC_PRIO, .len = 2, .offset = 2 }, + { .id = PFC_PRIO_0, .len = 2, .offset = 2, .mask = 0x0001 }, + { .id = PFC_PRIO_1, .len = 2, .offset = 2, .mask = 0x0002, .shift = 1 }, + { .id = PFC_PRIO_2, .len = 2, .offset = 2, .mask = 0x0004, .shift = 2 }, + { .id = PFC_PRIO_3, .len = 2, .offset = 2, .mask = 0x0008, .shift = 3 }, + { .id = PFC_PRIO_4, .len = 2, .offset = 2, .mask = 0x0010, .shift = 4 }, + { .id = PFC_PRIO_5, .len = 2, .offset = 2, .mask = 0x0020, .shift = 5 }, + { .id = PFC_PRIO_6, .len = 2, .offset = 2, .mask = 0x0040, .shift = 6 }, + { .id = PFC_PRIO_7, .len = 2, .offset = 2, .mask = 0x0080, .shift = 7 }, + { .id = PFC_TIME_0, .len = 2, .offset = 4, }, + { .id = PFC_TIME_1, .len = 2, .offset = 6, }, + { .id = PFC_TIME_2, .len = 2, .offset = 8, }, + { .id = PFC_TIME_3, .len = 2, .offset = 10, }, + { .id = PFC_TIME_4, .len = 2, .offset = 12, }, + { .id = PFC_TIME_5, .len = 2, .offset = 14, }, + { .id = PFC_TIME_6, .len = 2, .offset = 16, }, + { .id = PFC_TIME_7, .len = 2, .offset = 18, }, +}; + +static void pfc_header_init(struct proto_hdr *hdr) +{ + uint8_t eth_dst[6] = { 0x01, 0x80, 0xC2, 0x00, 0x00, 0x01 }; + + struct proto_hdr *lower; + + lower = proto_lower_default_add(hdr, PROTO_ETH); + proto_field_set_default_bytes(lower, ETH_DST_ADDR, eth_dst); + + proto_header_fields_add(hdr, pfc_fields, array_size(pfc_fields)); + proto_field_set_default_be16(hdr, PFC_OPCODE, 0x0101); +} + +static struct proto_ops pfc_proto_ops = { + .id = PROTO_PFC, + .layer = PROTO_L2, + .header_init = pfc_header_init, +}; + static struct proto_field vlan_fields[] = { /* TPID overlaps with Ethernet header and points to ether type */ { .id = VLAN_TPID, .len = 2, .offset = -2 }, @@ -194,6 +235,7 @@ void protos_l2_init(void) { proto_ops_register(ð_proto_ops); proto_ops_register(&pause_proto_ops); + proto_ops_register(&pfc_proto_ops); proto_ops_register(&vlan_proto_ops); proto_ops_register(&arp_proto_ops); proto_ops_register(&mpls_proto_ops); |