summaryrefslogtreecommitdiffstats
path: root/test/test_bond.py
AgeCommit message (Collapse)AuthorFilesLines
2020-09-02bonding: add bond_create2 API to include gso optionSteven Luong1-9/+15
gso option is available for the debug CLI version of bond create. This patch is to create a new API to have the corresponding option in the binary API. The old binary API bond_create is marked deprecated. Type: improvement Signed-off-by: Steven Luong <sluong@cisco.com> Change-Id: Id9501b8e6d267ae09e2b411957f181343da459c0
2020-07-28bonding: toggle bond admin state may cause hw link state to downSteven Luong1-2/+37
For bond interface, we set hardware link state to up initially when admin state is set to up. Thereafter, if we toggle the admin state to down and up, we require at least one active member in the bond prior to bringing up the hardware state which is inconsistent. The fix is to remove the unnecessary condition. This is a rework of the original proposed patch which is more complex and could be tricky to get it right https://gerrit.fd.io/r/c/vpp/+/27141 Type: fix Signed-off-by: Steven Luong <sluong@cisco.com> Change-Id: I65f4fee55299b5b4d5645f6e05e1be15b46b73cd
2020-07-28bonding lacp: replace slave string with memberSteven Luong1-30/+33
- Replace textual string slave with member except APIs. - For APIs, mark the existing APIs as deprecated and introduce new APIs - While introducing sw_bond_interface_dump, add the optional filter by sw_if_index and enhance the testcases to make use of it. Type: improvement Signed-off-by: Steven Luong <sluong@cisco.com> Change-Id: Ib6626c514e45350308aeeda0decb70f3aba2f63e
2020-05-29bonding: Revert adjust link stateMatthew Smith1-58/+1
This reverts commit 9121c415a91904be50071ec55143d9c89b2f1b91. Reason for revert: causes deadlock with multiple worker threads Type: fix Change-Id: Icf3d6d343d99d887abacbaa03acbf04d4628514e Signed-off-by: Matthew Smith <mgsmith@netgate.com>
2020-05-27bonding: adjust link state based on active slavesMatthew Smith1-1/+58
Type: improvement Bond link state is being maintained inconsistently. It is initially set to up. If the bond interface admin state is set to down, the link state is set to down. If the bond interface admin state is set to up, the link state is only set to up if there are active slave interfaces at that point. If slaves become active at some later time, it does not get updated. Its next chance to be updated is the next time the bond interface is set to admin up. To address this, do not set the link state to up after creating a bond. Adjust the link state as slave interfaces are attached or detached based on whether the bond is getting its first active slave or losing its last one. Unit test added to verify correct maintenance of link state. Change-Id: I31f17321f7f0e727e1ab1e01713423af6566dad9 Signed-off-by: Matthew Smith <mgsmith@netgate.com>
2020-02-05stats: fix state counter removalVladimir Isaev1-40/+41
Avoid using vec_del1() for directory vector to keep indexes valid all the time. There are state counters for each slave in LACP bond mode which can be dynamically created and removed. Vector index is used to access these counters. But also vec_del1() is used to remove counter from vector. This function changes the index of the last element, so after this we are unable to access ex-last element using old index. As a result it is not possible to add-del-add two interfaces to the LACP bond: DBGvpp# create bond mode lacp BondEthernet0 DBGvpp# create packet-generator interface pg1 DBGvpp# create packet-generator interface pg2 DBGvpp# bond add BondEthernet0 pg1 DBGvpp# bond add BondEthernet0 pg2 DBGvpp# bond del pg1 DBGvpp# bond del pg2 DBGvpp# bond add BondEthernet0 pg1 DBGvpp# bond add BondEthernet0 pg2 bond add: /if/lacp/1/3/partner-state is already register Type: fix Signed-off-by: Vladimir Isaev <visaev@netgate.com> Change-Id: I2c86e13905eefdef6233369cd4ab5c1b53d123bd
2019-12-17ip: Protocol Independent IP NeighborsNeale Ranns1-3/+3
Type: feature - ip-neighbour: generic neighbour handling; APIs, DBs, event handling, aging - arp: ARP protocol implementation - ip6-nd; IPv6 neighbor discovery implementation; separate ND, MLD, RA - ip6-link; manage link-local addresses - l2-arp-term; events separated from IP neighbours, since they are not the same. vnet retains just enough education to perform ND/ARP packet construction. arp and ip6-nd to be moved to plugins soon. Change-Id: I88dedd0006b299344f4c7024a0aa5baa6b9a8bbe Signed-off-by: Neale Ranns <nranns@cisco.com>
2019-11-15tests: Remove the unrequired VPP IP address/prefix class wrappersNeale Ranns1-2/+1
Type: refactor Change-Id: I204f3f8eebc5f5d5a377e91262f91c615fd00168 Signed-off-by: Neale Ranns <nranns@cisco.com>
2019-11-05misc: Fix python scripts shebang lineRenato Botelho do Couto1-1/+1
Type: fix Since CentOS 8, RPM build script doesn't accept '#!/usr/bin/env python' as a valid shebang line. It requires scripts to explicitly chose between python2 or python3. Change all to use python3 as suggested by Paul Vinciguerra. Depends-On: https://gerrit.fd.io/r/23170 Signed-off-by: Renato Botelho do Couto <renato@netgate.com> Change-Id: Ie72af9f60fd0609e07f05b70f8d96e738b2754d1
2019-10-21bonding: tests support python3Ole Troan1-2/+2
Type: fix Signed-off-by: Ole Troan <ot@cisco.com> Change-Id: I6cedb43af4437fdabce75f4a845629c5c2f96f6a
2019-09-06bonding: add weight support for active-backup modeSteven Luong1-6/+3
Not all interfaces have the same characteristics within the bonding group. For active-backup mode, we should do our best to select the slave that performs the best as the primary slave. We already did that by preferring the slave that is local numa. Sometimes, this is not enough. For example, when all are local numas, the selection is arbitrary. Some slave interfaces may have higher speed or better qos than the others. But this is hard to infer. One rule does not fit all. So we let the operator to optionally specify the weight for each slave interface. Our primary slave selection rule is now 1. biggest weight 2. is local numa 3. current primary slave (to avoid churn) 4. lowest sw_if_index (for deterministic behavior) This selection rule only applies to active-backup mode which only one slave is used for forwarding traffic until it becomes unreachable. At that time, the next "best" slave candidate is automatically promoted. The slaves are sorted according to the preference rule when they are up. So there is no need to find the next best candidate when the primary slave goes down. Another good thing about this rule is when the down slave comes back up, it is selected as the primary slave again unless there is indeed a "better" slave than this down slave that were added during that period. To set the weight for the slave interface, do this after the interface is enslaved set interface bond <interface-name> weight <value> Type: feature Signed-off-by: Steven Luong <sluong@cisco.com> Change-Id: I59ced6d20ce1dec532e667dbe1afd1b4243e04f9
2019-08-20api: Cleanup APIs interface.apiJakub Grajciar1-4/+4
Use of consistent API types for interface.api Type: fix Change-Id: I88206d7d0907cffd564031f73c9a996df2e5e21a Signed-off-by: Jakub Grajciar <jgrajcia@cisco.com>
2019-07-19bonding: add support for numa-only in lacp modeZhiyong Yang1-0/+1
If numa-only is set, Only slaves on local numa node transmit pkts if have at least one, otherwise the bond interface works as usual. CLI change: create bond mode lacp [load-balance { l2 | l23 | l34 } {numa-only}] [hw-addr <mac-address>] [id <if-id>] The new member "u8 numa_only;" is also added to bond_create_if_args_t. Type: feature Change-Id: Icdccedafb0738d8c9d4a5acce909ce562428c071 Signed-off-by: Zhiyong Yang <zhiyong.yang@intel.com>
2019-04-11Tests: Refactor tearDown show command logging, add lifecycle markers.Paul Vinciguerra1-2/+3
This change adds a consistent interface for adding test-specific show commands to log.txt. It also adds log markers for the execution of setUp[Class], tearDown[Class] in the logs. Change-Id: I7d42e396e594a59e866a7d55dac0af25548e657a Signed-off-by: Paul Vinciguerra <pvinci@vinciconsulting.com>
2019-04-10Tests Cleanup: Fix missing calls to setUpClass/tearDownClass.Paul Vinciguerra1-0/+4
Continuation/Part 2 of https://gerrit.fd.io/r/#/c/17092/ Change-Id: Id0122d84eaf2c05d29e5be63a594d5e528ee7c9a Signed-off-by: Paul Vinciguerra <pvinci@vinciconsulting.com>
2019-03-15Revert "API: Cleanup APIs interface.api"Ole Trøan1-4/+4
This reverts commit e63325e3ca03c847963863446345e6c80a2c0cfd. Allow time for CSIT to accommodate. Change-Id: I59435e4ab5e05e36a2796c3bf44889b5d4823cc2 Signed-off-by: ot@cisco.com
2019-03-15API: Cleanup APIs interface.apiJakub Grajciar1-4/+4
Use of consistent API types for interface.api Change-Id: Ieb54cebb4ac96b432a3f0b41596718aa2f34885b Signed-off-by: Jakub Grajciar <jgrajcia@cisco.com>
2019-03-11Tests: use self.assertNotIn().Paul Vinciguerra1-5/+5
Many tests use self.assertEqual(error.find("failed"), -1) Use self.assertNotIn("failed", error) to provide more meaningful errors such as AssertionError: 'Failed' not found in '' instead of 0 != -1. Change-Id: I670acdc977b788b2cedf94cfeafc12097781463f Signed-off-by: Paul Vinciguerra <pvinci@vinciconsulting.com>
2019-03-06test framework: vpp_papi_provider.py - further cleanupOle Troan1-3/+3
Part of further cleanups of this file. Removed most wrappers that don't have conflicting signature with message API. Change-Id: I6acd93d20291feb7731eb35ab2eb8c9f22f4632c Signed-off-by: Ole Troan <ot@cisco.com>
2018-12-18PAPI: Add MACAddress object wrapper for vl_api_mac_address_tOle Troan1-2/+2
Change the definition of vl_api_mac_address_t to an aliased type. Change-Id: I1434f316d0fad6a099592f39bceeb8faeaf1d134 Signed-off-by: Ole Troan <ot@cisco.com>
2018-12-10Test framework: StringIO fixes for Python3Ole Troan1-1/+1
Add 2/3 support to binarytomac and mactobinary and move to vpp_mac.py Change-Id: I3dc7e4a24486aee22140c781aae7e44e58935877 Signed-off-by: Ole Troan <ot@cisco.com>
2018-05-25bond: performance harvestingSteven1-11/+3
- hash is great. But it is a bit too slow for the DP. Use direct array indexing to quickly retrieve the slave interface. - the algorithm used by flow hash is great. But it is a bit too slow for the DP. Use l2_hash_hash() extracted from lb_hash.h which ECMP is using. It makes use of intrinsic crc32 instruction set. - shortcut modulo arithmetic when the operand is 2**x (where x up to 4) to avoid division instruction. - special case for link count == 1 in bond_tx_fn() - use clib_mem_unaligned to access data for the packet to avoid alignment error - Fix some typos for packet tracing. Change-Id: I8eae3ad497061c5473aa675ba894ee0211120d25 Signed-off-by: Steven <sluong@cisco.com>
2018-03-21bond: Add bonding driver and LACP protocolSteven1-0/+282
Add bonding driver to support creation of bond interface which composes of multiple slave interfaces. The slave interfaces could be physical interfaces, or just any virtual interfaces. For example, memif interfaces. The syntax to create a bond interface is create bond mode <lacp | xor | acitve-backup | broadcast | round-robin> To enslave an interface to the bond interface, enslave interface TenGigabitEthernet6/0/0 to BondEthernet0 Please see src/plugins/lacp/lacp_doc.md for more examples and additional options. LACP is a control plane protocol which manages and monitors the status of the slave interfaces. The protocol is part of 802.3ad standard. This patch implements LACPv1. LACPv2 is not supported. To enable LACP on the bond interface, specify "mode lacp" when the bond interface is created. The syntax to enslave a slave interface is the same as other bonding modes. Change-Id: I06581d3b87635972f9f0e1ec50b67560fc13e26c Signed-off-by: Steven <sluong@cisco.com>
uword zdata, uword * n_zdata_bits) { uword c, d, result, n_bits; uword explicit_end, implicit_end; result = 0; n_bits = 0; while (1) { c = first_set (coding); implicit_end = c == coding; explicit_end = (zdata & 1) & ~implicit_end; d = (zdata >> explicit_end) & (c - 1); if (explicit_end | implicit_end) { result += d; n_bits += min_log2 (c) + explicit_end; break; } n_bits += 1; result += c; coding ^= c; zdata >>= 1; } if (coding == 0) n_bits = BITS (uword); *n_zdata_bits = n_bits; return result; } uword zvec_encode (uword coding, uword data, uword * n_result_bits) { uword c, shift, result; uword explicit_end, implicit_end; /* Data must be in range. Note special coding == 0 would break for data - 1 <= coding. */ ASSERT (data <= coding - 1); shift = 0; while (1) { c = first_set (coding); implicit_end = c == coding; explicit_end = ((data & (c - 1)) == data); if (explicit_end | implicit_end) { uword t = explicit_end & ~implicit_end; result = ((data << t) | t) << shift; *n_result_bits = /* data bits */ (c == 0 ? BITS (uword) : min_log2 (c)) /* shift bits */ + shift + t; return result; } data -= c; coding ^= c; shift++; } /* Never reached. */ ASSERT (0); return ~0; } always_inline uword get_data (void *data, uword data_bytes, uword is_signed) { if (data_bytes == 1) return is_signed ? zvec_signed_to_unsigned (*(i8 *) data) : *(u8 *) data; else if (data_bytes == 2) return is_signed ? zvec_signed_to_unsigned (*(i16 *) data) : *(u16 *) data; else if (data_bytes == 4) return is_signed ? zvec_signed_to_unsigned (*(i32 *) data) : *(u32 *) data; else if (data_bytes == 8) return is_signed ? zvec_signed_to_unsigned (*(i64 *) data) : *(u64 *) data; else { os_panic (); return ~0; } } always_inline void put_data (void *data, uword data_bytes, uword is_signed, uword x) { if (data_bytes == 1) { if (is_signed) *(i8 *) data = zvec_unsigned_to_signed (x); else *(u8 *) data = x; } else if (data_bytes == 2) { if (is_signed) *(i16 *) data = zvec_unsigned_to_signed (x); else *(u16 *) data = x; } else if (data_bytes == 4) { if (is_signed) *(i32 *) data = zvec_unsigned_to_signed (x); else *(u32 *) data = x; } else if (data_bytes == 8) { if (is_signed) *(i64 *) data = zvec_unsigned_to_signed (x); else *(u64 *) data = x; } else { os_panic (); } } always_inline uword * zvec_encode_inline (uword * zvec, uword * zvec_n_bits, uword coding, void *data, uword data_stride, uword n_data, uword data_bytes, uword is_signed) { uword i; i = *zvec_n_bits; while (n_data >= 1) { uword d0, z0, l0; d0 = get_data (data + 0 * data_stride, data_bytes, is_signed); data += 1 * data_stride; n_data -= 1; z0 = zvec_encode (coding, d0, &l0); zvec = clib_bitmap_set_multiple (zvec, i, z0, l0); i += l0; } *zvec_n_bits = i; return zvec; } #define _(TYPE,IS_SIGNED) \ uword * zvec_encode_##TYPE (uword * zvec, \ uword * zvec_n_bits, \ uword coding, \ void * data, \ uword data_stride, \ uword n_data) \ { \ return zvec_encode_inline (zvec, zvec_n_bits, \ coding, \ data, data_stride, n_data, \ /* data_bytes */ sizeof (TYPE), \ /* is_signed */ IS_SIGNED); \ } _(u8, /* is_signed */ 0); _(u16, /* is_signed */ 0); _(u32, /* is_signed */ 0); _(u64, /* is_signed */ 0); _(i8, /* is_signed */ 1); _(i16, /* is_signed */ 1); _(i32, /* is_signed */ 1); _(i64, /* is_signed */ 1); #undef _ always_inline uword coding_max_n_bits (uword coding) { uword n_bits; (void) zvec_decode (coding, 0, &n_bits); return n_bits; } always_inline void zvec_decode_inline (uword * zvec, uword * zvec_n_bits, uword coding, void *data, uword data_stride, uword n_data, uword data_bytes, uword is_signed) { uword i, n_max; i = *zvec_n_bits; n_max = coding_max_n_bits (coding); while (n_data >= 1) { uword d0, z0, l0; z0 = clib_bitmap_get_multiple (zvec, i, n_max); d0 = zvec_decode (coding, z0, &l0); i += l0; put_data (data + 0 * data_stride, data_bytes, is_signed, d0); data += 1 * data_stride; n_data -= 1; } *zvec_n_bits = i; } #define _(TYPE,IS_SIGNED) \ void zvec_decode_##TYPE (uword * zvec, \ uword * zvec_n_bits, \ uword coding, \ void * data, \ uword data_stride, \ uword n_data) \ { \ return zvec_decode_inline (zvec, zvec_n_bits, \ coding, \ data, data_stride, n_data, \ /* data_bytes */ sizeof (TYPE), \ /* is_signed */ IS_SIGNED); \ } _(u8, /* is_signed */ 0); _(u16, /* is_signed */ 0); _(u32, /* is_signed */ 0); _(u64, /* is_signed */ 0); _(i8, /* is_signed */ 1); _(i16, /* is_signed */ 1); _(i32, /* is_signed */ 1); _(i64, /* is_signed */ 1); #undef _ /* Compute number of bits needed to encode given histogram. */ static uword zvec_coding_bits (uword coding, uword * histogram_counts, uword min_bits) { uword n_type_bits, n_bits; uword this_count, last_count, max_count_index; uword i, b, l; n_bits = 0; n_type_bits = 1; last_count = 0; max_count_index = vec_len (histogram_counts) - 1; /* Coding is not large enough to encode given data. */ if (coding <= max_count_index) return ~0; i = 0; while (coding != 0) { b = first_set (coding); l = min_log2 (b); i += b; this_count = histogram_counts[i > max_count_index ? max_count_index : i - 1]; /* No more data to encode? */ if (this_count == last_count) break; /* Last coding is i 0 ... 0 so we don't need an extra type bit. */ if (coding == b) n_type_bits--; n_bits += (this_count - last_count) * (n_type_bits + l); /* This coding cannot be minimal: so return. */ if (n_bits >= min_bits) return ~0; last_count = this_count; coding ^= b; n_type_bits++; } return n_bits; } uword _zvec_coding_from_histogram (void *histogram, uword histogram_len, uword histogram_elt_count_offset, uword histogram_elt_bytes, uword max_value_to_encode, zvec_coding_info_t * coding_return) { uword coding, min_coding; uword min_coding_bits, coding_bits; uword i, n_bits_set, total_count; uword *counts; zvec_histogram_count_t *h_count = histogram + histogram_elt_count_offset; if (histogram_len < 1) { coding_return->coding = 0; coding_return->min_coding_bits = 0; coding_return->n_data = 0; coding_return->n_codes = 0; coding_return->ave_coding_bits = 0; return 0; } total_count = 0; counts = vec_new (uword, histogram_len); for (i = 0; i < histogram_len; i++) { zvec_histogram_count_t this_count = h_count[0]; total_count += this_count; counts[i] = total_count; h_count = (zvec_histogram_count_t *) ((void *) h_count + histogram_elt_bytes); } min_coding = 0; min_coding_bits = ~0; { uword base_coding = max_value_to_encode != ~0 ? (1 + max_value_to_encode) : vec_len (counts); uword max_coding = max_pow2 (2 * base_coding); for (n_bits_set = 1; n_bits_set <= 8; n_bits_set++) { for (coding = pow2_mask (n_bits_set); coding < max_coding; coding = next_with_same_number_of_set_bits (coding)) { coding_bits = zvec_coding_bits (coding, counts, min_coding_bits); if (coding_bits >= min_coding_bits) continue; min_coding_bits = coding_bits; min_coding = coding; } } } if (coding_return) { coding_return->coding = min_coding; coding_return->min_coding_bits = min_coding_bits; coding_return->n_data = total_count; coding_return->n_codes = vec_len (counts); coding_return->ave_coding_bits = (f64) min_coding_bits / (f64) total_count; } vec_free (counts); return min_coding; } u8 * format_zvec_coding (u8 * s, va_list * args) { zvec_coding_info_t *c = va_arg (*args, zvec_coding_info_t *); return format (s, "zvec coding 0x%x, %d elts, %d codes, %d bits total, %.4f ave bits/code", c->coding, c->n_data, c->n_codes, c->min_coding_bits, c->ave_coding_bits); } /* * fd.io coding-style-patch-verification: ON * * Local Variables: * eval: (c-set-style "gnu") * End: */