summaryrefslogtreecommitdiffstats
AgeCommit message (Collapse)AuthorFilesLines
2018-12-06nsim: add packet loss simulation, docsDave Barach8-9/+170
Change-Id: Ic9747541aad8148ebf7d520b525b99c4cc3961f3 Signed-off-by: Dave Barach <dave@barachs.net>
2018-12-06DOC-ONLY: MFIB documentationNeale Ranns4-5/+96
Change-Id: I8c7277584f231dd4732a4c6013e156a5bb821f41 Signed-off-by: Neale Ranns <nranns@cisco.com>
2018-12-06API: Change ip4_address and ip6_address to use type alias.Ole Troan23-144/+94
Change-Id: Id8669bbadd1d6b2054865a310a654e9b38d1667d Signed-off-by: Ole Troan <ot@cisco.com>
2018-12-06Update japi to support type aliasesMichal Cmarada5-32/+126
Change-Id: I6d6068d641d4c91e5c5b52eefb898affc5c0d2c0 Signed-off-by: Michal Cmarada <mcmarada@cisco.com>
2018-12-06MFIB; CLI improvementsNeale Ranns2-7/+37
Change-Id: I7cf3ae8c10dd584e8bc234a3253bea3c5a2d105a Signed-off-by: Neale Ranns <nranns@cisco.com>
2018-12-05Fix gcc-8 compile issues in string_test.cDave Barach1-1/+35
gcc-8 flunks a certain number of tests at compile time, so conditionally disable (negative) tests which won't even compile. Change-Id: Id7e85f38bc371623972efa6e2c8f9ee4717f5ff5 Signed-off-by: Dave Barach <dave@barachs.net>
2018-12-05Improve strncpy_s src/dst overlap checkDave Barach1-2/+12
Let m = user estimate of the (max) src string length, low = smaller address of (src, dst), hi = larger address (src, dst). if (low + (m - 1) >= hi), we have a *potential* overlapping copy which is not allowed. Before we declare overlap - and return an error - retry the check with m = actual src string length. The resulting "test string" failure affected aarch64 (only) because of differences in test code stack variable placement / alignment. Change-Id: I2931d1ce2c61af3d3880075b033d2a4c4e421f09 Signed-off-by: Dave Barach <dave@barachs.net>
2018-12-05bapi: add options to have vpp cleanup client registrationFlorin Coras5-25/+36
A client can send a memclnt delete message and ask vpp to cleanup the shared memory queue. Obviously, in this case no delete reply is sent back to the client. Change-Id: I9c8375093f8607680ad498a6bed0690ba02a7c3b Signed-off-by: Florin Coras <fcoras@cisco.com>
2018-12-05ldp: avoid calling write in destructorsFlorin Coras2-5/+8
Change-Id: Ia9c3d7a68a23dc4ab3be06f88fdfb053db422372 Signed-off-by: Florin Coras <fcoras@cisco.com>
2018-12-05session/tcp: postpone cleanup on resetFlorin Coras8-13/+39
Change-Id: I45fd7538853f84c6c8bf804cc20acbc9601db3ba Signed-off-by: Florin Coras <fcoras@cisco.com>
2018-12-05DOC ONLY: describe dispatch pcap tracingDave Barach3-2/+263
and wireshark dissection of these traces. Change-Id: I61029fd20d6d5f6c40638e3ea9223f2354abedba Signed-off-by: Dave Barach <dave@barachs.net>
2018-12-05stat-client: pass names as const stringsNeale Ranns2-4/+4
Change-Id: I8debcd078b733ed19c4b46ea41a2a150d816724c Signed-off-by: Neale Ranns <nranns@cisco.com>
2018-12-05VOM: interface event structNeale Ranns7-5/+72
Change-Id: If133829ba4db2da1c9c20bfbbdfc6df6276efa10 Signed-off-by: Neale Ranns <nranns@cisco.com>
2018-12-05VPP-1508: Python3 compatible printPaul Vinciguerra6-24/+23
Change-Id: I7d2597f2a9589f2463e213da6ca67b1b7593095e Signed-off-by: Paul Vinciguerra <pvinci@vinciconsulting.com>
2018-12-05VPP-1508 python3 tests: xrangePaul Vinciguerra2-31/+34
xrange is not supported. Use six.range. py27 runtests: commands[5] | stestr --test-path ./test run --slowest test_ip4 test_ip6 ============================================================================== IPv4 disabled ============================================================================== ============================================================================== ICMP Echo Test Case ============================================================================== {0} test.test_ip4.TestIPDisabled.test_ip_disabled [5.256819s] ... ok 07:24:41,902 Couldn't stat : /tmp/vpp-unittest-TestICMPEcho-hU4IsB/stats.sock {1} test.test_ip4.TestICMPEcho.test_icmp_echo [0.367035s] ... ok ============================================================================== IPv4 Deaggregate Routes ============================================================================== ============================================================================== IPv4 Input Exceptions ============================================================================== 07:24:47,314 Couldn't stat : /tmp/vpp-unittest-TestIPDeag-eE1VgC/stats.sock {1} test.test_ip4.TestIPDeag.test_ip_deag [5.895646s] ... ok {0} test.test_ip4.TestIPInput.test_ip_input [5.819001s] ... ok ============================================================================== IPv4 longest Prefix Match ... output truncated ... ============================== Failed 4 tests - output below: ============================== test.test_ip4.TestIPv4FibCrud.test_3_add_new_routes --------------------------------------------------- Captured traceback: ~~~~~~~~~~~~~~~~~~ Traceback (most recent call last): File "/vpp/test/test_ip4.py", line 509, in test_3_add_new_routes self.deleted_routes.remove(x) ValueError: list.remove(x): x not in list test.test_ip4.TestIPv4FibCrud.test_2_del_routes ----------------------------------------------- Captured traceback: ~~~~~~~~~~~~~~~~~~ Traceback (most recent call last): File "/vpp/test/test_ip4.py", line 478, in test_2_del_routes self.configured_routes.remove(x) ValueError: list.remove(x): x not in list test.test_ip4_vrf_multi_instance.TestIp4VrfMultiInst.test_ip4_vrf_03 -------------------------------------------------------------------- Captured traceback: ~~~~~~~~~~~~~~~~~~ Traceback (most recent call last): File "/vpp/test/test_ip4_vrf_multi_instance.py", line 465, in test_ip4_vrf_03 self.create_vrf_and_assign_interfaces(1) File "/vpp/test/test_ip4_vrf_multi_instance.py", line 189, in create_vrf_and_assign_interfaces pg_if.set_table_ip4(vrf_id) File "/vpp/test/vpp_interface.py", line 322, in set_table_ip4 self.sw_if_index, 0, self.ip4_table_id) File "/vpp/test/vpp_papi_provider.py", line 264, in sw_interface_set_table 'vrf_id': table_id}) File "/vpp/test/vpp_papi_provider.py", line 196, in api raise UnexpectedApiReturnValueError(msg) test.vpp_papi_provider.UnexpectedApiReturnValueError: API call failed, expected 0 return value instead of -114 in sw_interface_set_table_reply(_0=91, context=1007, retval=-114) test.test_ip4_vrf_multi_instance.TestIp4VrfMultiInst.test_ip4_vrf_02 -------------------------------------------------------------------- Captured traceback: ~~~~~~~~~~~~~~~~~~ Traceback (most recent call last): File "/vpp/test/test_ip4_vrf_multi_instance.py", line 445, in test_ip4_vrf_02 self.reset_vrf_and_remove_from_vrf_list(1) File "/vpp/test/test_ip4_vrf_multi_instance.py", line 208, in reset_vrf_and_remove_from_vrf_list self.vapi.reset_fib(vrf_id, is_ipv6=0) File "/vpp/test/vpp_papi_provider.py", line 1137, in reset_fib 'is_ipv6': is_ipv6, File "/vpp/test/vpp_papi_provider.py", line 196, in api raise UnexpectedApiReturnValueError(msg) test.vpp_papi_provider.UnexpectedApiReturnValueError: API call failed, expected 0 return value instead of -3 in reset_fib_reply(_0=259, context=1198, retval=-3) ====== Totals ====== Ran: 57 tests in 266.0000 sec. - Passed: 53 - Skipped: 0 - Expected Fail: 0 - Unexpected Success: 0 - Failed: 4 Sum of execute time for each test: 157.3925 sec. ============== Worker Balance ============== - Worker 0 (29 tests) => 0:03:52.608995 - Worker 1 (28 tests) => 0:04:08.615473 Test id Runtime (s) --------------------------------------------- ----------- test.test_ip_mcast.TestIPMcast.test_ip6_mcast 8.535 test.test_ip4.TestIPPunt.test_ip_punt 8.082 test.test_ip6.TestIP6Punt.test_ip_punt 6.582 test.test_ip6.TestIPDeag.test_ip_deag 6.175 test.test_ip6.TestIPv6.test_ns 6.171 test.test_ip4.TestIPDeag.test_ip_deag 5.896 test.test_ip6.TestIPv6.test_fib 5.846 test.test_ip4.TestIPInput.test_ip_input 5.819 test.test_ip6.TestIPv6.test_rs 5.737 test.test_ip4.TestIPv4.test_fib 5.267 ERROR: InvocationError for command '/vpp/.tox/py27/bin/stestr --test-path ./test run --slowest test_ip4 test_ip6' (exited with code 1) ______________________________________________________ summary ______________________________________________________ ERROR: py27: commands failed Change-Id: Id9f6ecb4897386f790d82ab908963e4971a3aac8 Signed-off-by: Paul Vinciguerra <pvinci@vinciconsulting.com>
2018-12-04ipsec: simplify bumping counters - cosmetic changeKlement Sekera4-116/+35
Change-Id: Ibb55427ed49d0277854a352922c6c4bb007bf072 Signed-off-by: Klement Sekera <ksekera@cisco.com>
2018-12-04vcl: cleanup children that use _exit()Florin Coras5-32/+154
Change-Id: Ia56c2698adb0ea7811203844dc4db10e121fbc42 Signed-off-by: Florin Coras <fcoras@cisco.com>
2018-12-04Add VNET_BUFFER_F_AVAIL definitionsDave Barach2-1/+26
Add a check to make sure that the vlib and vnet buffer flag bit definitions do not overlap. The VNET_BUFFER_F_AVAIL1...8 definitions allow out-of-tree codes to: #define VNET_BUFFER_F_MY_USECASE VNET_BUFFER_F_AVAIL1 and so on. This avoids introducing irrelevant and/or proprietary bit definitions into vnet/buffer.h, and hopefully minimizes merge pain for everyone involved. Change-Id: I5be4f61dceb81b5bfca005f6d609ade074af205b Signed-off-by: Dave Barach <dave@barachs.net>
2018-12-04vcl: test refactor and improvementsFlorin Coras7-277/+106
Change-Id: I92f415bf253d6e051ec9d94ebeb98f081b2a0293 Signed-off-by: Florin Coras <fcoras@cisco.com>
2018-12-04MPLS: buffer over-run with incorrectly init'd vector. fix VAT dumpNeale Ranns2-3/+3
Change-Id: Ifdbb4c4cffd90c4ec8b39513d284ebf7be39eca5 Signed-off-by: Neale Ranns <nranns@cisco.com>
2018-12-03Format vnet_buffer_t l2 feature bitmapDave Barach1-0/+8
Change-Id: Iad3120ab6466b77875efc89ccb49f6d22e36e62a Signed-off-by: Dave Barach <dave@barachs.net>
2018-12-03API: C generator improve symbol protection.Ole Troan1-2/+2
Change-Id: I9ef042c06c12892a2bbc23cc3bac6891c77e3feb Signed-off-by: Ole Troan <ot@cisco.com>
2018-12-03vcl: handle worker process exitFlorin Coras6-22/+56
Change-Id: Ife05d25fd736ae3064f01e974e5aecc5b48de924 Signed-off-by: Florin Coras <fcoras@cisco.com>
2018-12-03svm: use explicit svm fifo segment main for slavesFlorin Coras8-62/+76
Change-Id: Id39d64bf1b49345a3dc31c63360569212aba6865 Signed-off-by: Florin Coras <fcoras@cisco.com>
2018-12-03Add UDP encap flagFilip Tehlar2-0/+3
Change-Id: Ic6a8b9aaec7e5dee4fb1971168988dbe4f931f86 Signed-off-by: Filip Tehlar <ftehlar@cisco.com>
2018-12-03move [m]fib and BIER tests to unittest pluginNeale Ranns5-3/+3
Change-Id: I9d2f52e756363df011026773bfffa838a557313f Signed-off-by: Neale Ranns <nranns@cisco.com>
2018-12-03Copying QoS Bits when fragmented, so that marking can happen properly also ↵Vijayabhaskar Katamreddy1-0/+7
cleaning up some unused code Change-Id: I1558eec79af173e5cdcc769d7c3909039403eed8 Signed-off-by: Vijayabhaskar Katamreddy <vkatamre@cisco.com>
2018-12-02IPSEC-AH: anti-replay testingNeale Ranns6-36/+120
Change-Id: Ia5d45db73e4bdb32214ed4f365d5eec8e28115f3 Signed-off-by: Neale Ranns <nranns@cisco.com>
2018-12-02vppinfra: c11 safe string functionsSteven4-25/+2682
Add memcmp_s, strcmp_s, strncmp_s, strcpy_s, strncpy_s, strcat_s, strncat_s, strtok_s, strnlen_s, and strstr_s C11 safe string API. For migrating extant unsafe API, add also the corresponding macro version of each safe API, clib_memcmp, clib_strcmp, etc. In general, the benefits of the safe string APIs are to provide null pointer checks, add additional argument to specify the string length of the passed string rather than relying on the null terminated character, and src/dest overlap checking for the the string copy operations. The macro version of the API takes the same number of arguments as the unsafe API to provide easy migration. However, it does not usually provide the full aformentioned benefits. In some cases, it is necessary to move to the safe API rather than using the macro in order to avoid some unpredictable problems such as accessing memory beyond what it is intended due to the lack of the passed string length. dbarach: add a "make test" vector, and a doxygen file header cookie. Change-Id: I5cd79b8928dcf76a79bf3f0b8cbc1a8f24942f4c Signed-off-by: Steven <sluong@cisco.com> Signed-off-by: Dave Barach <dave@barachs.net>
2018-12-01ip_reassembly_enable_disable reply handler name is changed.Chore1-1/+1
Change-Id: I89be597376690bb75c4347bcfc1c6c3d27c4034c Signed-off-by: Chore <s3m2e1.6star@gmail.com>
2018-12-01dpdk-ipsec-mempool: allocate from dpdk mem specified by socket-mem in ↵Kingwel Xie1-22/+17
startup.conf otherwise, these pools will occupy an entire huge page for each even they are very small. Change-Id: I08919714de9b6cd4b8dddb546ca54364b56ec99f Signed-off-by: Kingwel Xie <kingwel.xie@ericsson.com>
2018-12-01Delete shared memory segment files when vpp startsDave Barach3-3/+23
Should have been done this way years ago. My bad. Change-Id: Ic7bf937fb6c4dc5c1b6ae64f2ecf8608b62e7039 Signed-off-by: Dave Barach <dave@barachs.net>
2018-11-30Fix L2BD arp termination Test CasePaul Vinciguerra1-3/+7
============================================================================== L2BD arp termination Test Case ============================================================================== 12:02:21,850 Couldn't stat : /tmp/vpp-unittest-TestL2bdArpTerm-_h44qo/stats.sock L2BD arp term - add 5 hosts, verify arp responses OK L2BD arp term - delete 3 hosts, verify arp responses OK L2BD arp term - recreate BD1, readd 3 hosts, verify arp responses OK L2BD arp term - 2 IP4 addrs per host OK L2BD arp term - create and update 10 IP4-mac pairs OK L2BD arp/ND term - hosts with both ip4/ip6 OK L2BD ND term - Add and Del hosts, verify ND replies OK L2BD ND term - Add and update IP+mac, verify ND replies OK L2BD arp term - send garps, verify arp event reports OK L2BD arp term - send duplicate garps, verify suppression OK L2BD arp term - disable ip4 arp events,send garps, verify no events OK L2BD ND term - send NS packets verify reports OK L2BD ND term - send duplicate ns, verify suppression OK L2BD ND term - disable ip4 arp events,send ns, verify no events OK ============================================================================== TEST RESULTS: Scheduled tests: 14 Executed tests: 14 Passed tests: 14 ============================================================================== Test run was successful Change-Id: I6bb1ced11b88080ffaa845d22b0bc471c4f91683 Signed-off-by: Paul Vinciguerra <pvinci@vinciconsulting.com>
2018-11-30Metadata / opaque formatting belongs in vppDave Barach10-862/+526
VPP graph dispatch trace record description: 0 1 2 3 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | Major Version | Minor Version | NStrings | ProtoHint | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | Buffer index (big endian) | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ + VPP graph node name ... ... | NULL octet | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | Buffer Metadata ... ... | NULL octet | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | Buffer Opaque ... ... | NULL octet | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | Buffer Opaque 2 ... ... | NULL octet | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | VPP ASCII packet trace (if NStrings > 4) | NULL octet | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ | Packet data (up to 16K) | +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+ Graph dispatch records comprise a version stamp, an indication of how many NULL-terminated strings will follow the record header, and a protocol hint. The buffer index allows downstream consumers of these data to easily filter/track single packets as they traverse the forwarding graph. FWIW, the 32-bit buffer index is stored in big endian format. As of this writing, major version = 1, minor version = 0. Nstrings will be either 4 or 5. Here is the current set of protocol hints: typedef enum { VLIB_NODE_PROTO_HINT_NONE = 0, VLIB_NODE_PROTO_HINT_ETHERNET, VLIB_NODE_PROTO_HINT_IP4, VLIB_NODE_PROTO_HINT_IP6, VLIB_NODE_PROTO_HINT_TCP, VLIB_NODE_PROTO_HINT_UDP, VLIB_NODE_N_PROTO_HINTS, } vlib_node_proto_hint_t; Example: VLIB_NODE_PROTO_HINT_IP6 means that the first octet of packet data SHOULD be 0x60, and should begin an ipv6 packet header. Change-Id: Idf310bad80cc0e4207394c80f18db5f77c378741 Signed-off-by: Dave Barach <dave@barachs.net>
2018-11-30coding-style: missing coding-style-patchKingwel Xie1-168/+200
fix coding style Change-Id: I458d81fa80c509b71edb2021468a89715cb32ae3 Signed-off-by: Kingwel Xie <kingwel.xie@ericsson.com>
2018-11-30IPSEC-AH: fix packet dropNeale Ranns2-7/+0
Change-Id: I45b97cfd0c3785bfbf6d142d362bd3d4d56bae00 Signed-off-by: Neale Ranns <nranns@cisco.com>
2018-11-30make test: print TEST= values for failed testsKlement Sekera1-4/+4
This prints the format strings required to rerun the failed test next to the failures. Change-Id: I973d8ae025f026129826356bce265b1e70086d2f Signed-off-by: Klement Sekera <ksekera@cisco.com>
2018-11-30GBP: UT more robust for endpoint timeoutNeale Ranns1-16/+17
Change-Id: Ib78c747fa3e304f7158f8182d060e11e4e778400 Signed-off-by: Neale Ranns <nranns@cisco.com>
2018-11-30vcl/test: increase wait time before connectFlorin Coras1-10/+66
Also split tests into smaller groups Change-Id: I35809607b7a59029606bb34b90cfeffd1985fe60 Signed-off-by: Florin Coras <fcoras@cisco.com>
2018-11-30vcl: wait for segments with segment handleFlorin Coras10-71/+154
Instead of waiting for notification from binary api. Change-Id: I5ecab857d6bcdbed62d6bb06709570c4cf6b19ea Signed-off-by: Florin Coras <fcoras@cisco.com>
2018-11-30API: Add dependency on compiler for API .json and .api generation.Ole Troan1-2/+2
Add dependency on the API compiler, so that builds can deal with changes to the tool. Change-Id: I2587235fefa93a69955495870d49f36b4203bfea Signed-off-by: Ole Troan <ot@cisco.com>
2018-11-30Tests: Fix traceback.Paul Vinciguerra3-129/+182
self.assertTrue(packet.haslayer(msg_type)) File "/usr/lib/python2.7/unittest/case.py", line 422, in assertTrue raise self.failureException(msg) AssertionError: 0 is not true * Scapy packet.haslayer() returns 1 or 0. Replace with assertEqual(packet.haslayer(), 1) to fix tracebacks. * Scapy has multiple layers called TCP/UDP Specify the module name to prevent namespace collisions. * Remove duplicate import. Change-Id: I600f9f330075cd40e1da50f8b2ceb24f645f2c20 Signed-off-by: Paul Vinciguerra <pvinci@vinciconsulting.com>
2018-11-30session: segment handle in accept/connect notificationsFlorin Coras19-50/+122
Change-Id: I03884b6cde9d4c38ae13d1994fd8d37d44016ef0 Signed-off-by: Florin Coras <fcoras@cisco.com>
2018-11-30session: use endpt fib index if app in default nsJohn Lo1-1/+6
Change-Id: Icf1408c50a6438c81e16033e83b2a76ce6eb0166 Signed-off-by: John Lo <loj@cisco.com> Signed-off-by: Florin Coras <fcoras@cisco.com>
2018-11-29GRE: UT uses new pkt send functionsNeale Ranns1-108/+23
Change-Id: Icb1767b31421aa1e427498adc82d79307835ee1a Signed-off-by: Neale Ranns <nranns@cisco.com>
2018-11-29make test: verify packet counters in ipsec testsKlement Sekera1-0/+16
Change-Id: Ia15b40f9d91daf3a325cb8d707b22a1dbc68d9cc Signed-off-by: Klement Sekera <ksekera@cisco.com>
2018-11-29do not opttimize graph node functions in debug buildsDamjan Marion1-1/+1
Change-Id: I5b4cd419d317381a06e7e6d703373959f4bbd97b Signed-off-by: Damjan Marion <damarion@cisco.com>
2018-11-29DHCP: Initialise trace for copied buffers.Ole Troan2-0/+4
DHCP tests failed intermittantly with a core dump. Let's see if this fixes it. Change-Id: I42829a2c7e7f5a9a6775330d37bf972ff0008210 Signed-off-by: Ole Troan <ot@cisco.com>
2018-11-29vpp_papi: Add custom exceptions.Paul Vinciguerra6-54/+121
This patchset adds and raises the following custom exception classes: * class VPPApiError(Exception): * class VPPNotImplementedError(NotImplementedError): * class VPPIOError(IOError): * class VPPRuntimeError(RuntimeError): * class VPPValueError(ValueError): * class VPPSerializerValueError(ValueError): * class VPPStatsIOError(IOError): * class VPPStatsClientLoadError(RuntimeError): * class VppTransportShmemIOError(IOError): * class VppTransportSocketIOError(IOError) Change-Id: Ia40900fd2dcef148d01125d6c691329fc666901e Signed-off-by: Paul Vinciguerra <pvinci@vinciconsulting.com>
2018-11-29vpp_papi: Fix: raise NotImplemented.Paul Vinciguerra1-1/+1
'raise NotImplemented' should be 'raise NotImplementedError'. NotImplemented is not part of the Exception heirarchy. Change-Id: I7fb647f1d56e689fafa2cd9a5566da826def072b Signed-off-by: Paul Vinciguerra <pvinci@vinciconsulting.com>
lass="p">); CLIB_PREFETCH (b3->data, sizeof (srp_header_t), LOAD); } bi0 = from[0]; bi1 = from[1]; to_next[0] = bi0; to_next[1] = bi1; from += 2; to_next += 2; n_left_to_next -= 2; n_left_from -= 2; b0 = vlib_get_buffer (vm, bi0); b1 = vlib_get_buffer (vm, bi1); s0 = (void *) (b0->data + b0->current_data); s1 = (void *) (b1->data + b1->current_data); /* Data packets are always assigned to side A (outer ring) interface. */ sw_if_index0 = vnet_buffer (b0)->sw_if_index[VLIB_RX]; sw_if_index1 = vnet_buffer (b1)->sw_if_index[VLIB_RX]; hi0 = vnet_get_sup_hw_interface (vnm, sw_if_index0); hi1 = vnet_get_sup_hw_interface (vnm, sw_if_index1); si0 = pool_elt_at_index (sm->interface_pool, hi0->hw_instance); si1 = pool_elt_at_index (sm->interface_pool, hi1->hw_instance); sw_if_index0 = (s0->mode == SRP_MODE_data ? si0->rings[SRP_RING_OUTER].sw_if_index : sw_if_index0); sw_if_index1 = (s1->mode == SRP_MODE_data ? si1->rings[SRP_RING_OUTER].sw_if_index : sw_if_index1); vnet_buffer (b0)->sw_if_index[VLIB_RX] = sw_if_index0; vnet_buffer (b1)->sw_if_index[VLIB_RX] = sw_if_index1; d0 = srp_input_disposition_by_mode + s0->mode; d1 = srp_input_disposition_by_mode + s1->mode; next0 = d0->next_index; next1 = d1->next_index; error0 = d0->error; error1 = d1->error; vlib_buffer_advance (b0, d0->buffer_advance); vlib_buffer_advance (b1, d1->buffer_advance); b0->error = node->errors[error0]; b1->error = node->errors[error1]; vlib_validate_buffer_enqueue_x2 (vm, node, next_index, to_next, n_left_to_next, bi0, bi1, next0, next1); } while (n_left_from > 0 && n_left_to_next > 0) { u32 bi0, sw_if_index0; vlib_buffer_t * b0; u8 next0, error0; srp_header_t * s0; srp_input_disposition_t * d0; srp_interface_t * si0; vnet_hw_interface_t * hi0; bi0 = from[0]; to_next[0] = bi0; from += 1; to_next += 1; n_left_to_next -= 1; n_left_from -= 1; b0 = vlib_get_buffer (vm, bi0); s0 = (void *) (b0->data + b0->current_data); /* Data packets are always assigned to side A (outer ring) interface. */ sw_if_index0 = vnet_buffer (b0)->sw_if_index[VLIB_RX]; hi0 = vnet_get_sup_hw_interface (vnm, sw_if_index0); si0 = pool_elt_at_index (sm->interface_pool, hi0->hw_instance); sw_if_index0 = (s0->mode == SRP_MODE_data ? si0->rings[SRP_RING_OUTER].sw_if_index : sw_if_index0); vnet_buffer (b0)->sw_if_index[VLIB_RX] = sw_if_index0; d0 = srp_input_disposition_by_mode + s0->mode; next0 = d0->next_index; error0 = d0->error; vlib_buffer_advance (b0, d0->buffer_advance); b0->error = node->errors[error0]; vlib_validate_buffer_enqueue_x1 (vm, node, next_index, to_next, n_left_to_next, bi0, next0); } vlib_put_next_frame (vm, node, next_index, n_left_to_next); } return from_frame->n_vectors; } static char * srp_error_strings[] = { #define _(f,s) s, foreach_srp_error #undef _ }; static vlib_node_registration_t srp_input_node = { .function = srp_input, .name = "srp-input", /* Takes a vector of packets. */ .vector_size = sizeof (u32), .n_errors = SRP_N_ERROR, .error_strings = srp_error_strings, .n_next_nodes = SRP_INPUT_N_NEXT, .next_nodes = { [SRP_INPUT_NEXT_ERROR] = "error-drop", [SRP_INPUT_NEXT_ETHERNET_INPUT] = "ethernet-input", [SRP_INPUT_NEXT_CONTROL] = "srp-control", }, .format_buffer = format_srp_header_with_length, .format_trace = format_srp_input_trace, .unformat_buffer = unformat_srp_header, }; static uword srp_topology_packet (vlib_main_t * vm, u32 sw_if_index, u8 ** contents) { vnet_main_t * vnm = vnet_get_main(); vnet_hw_interface_t * hi = vnet_get_sup_hw_interface (vnm, sw_if_index); srp_topology_header_t * t; srp_topology_mac_binding_t * mb; u32 nb, nmb; t = (void *) *contents; nb = clib_net_to_host_u16 (t->n_bytes_of_data_that_follows); nmb = (nb - sizeof (t->originator_address)) / sizeof (mb[0]); if (vec_len (*contents) < sizeof (t[0]) + nmb * sizeof (mb[0])) return SRP_ERROR_TOPOLOGY_BAD_LENGTH; /* Fill in our source MAC address. */ clib_memcpy (t->ethernet.src_address, hi->hw_address, vec_len (hi->hw_address)); /* Make space for our MAC binding. */ vec_resize (*contents, sizeof (srp_topology_mac_binding_t)); t = (void *) *contents; t->n_bytes_of_data_that_follows = clib_host_to_net_u16 (nb + sizeof (mb[0])); mb = t->bindings + nmb; mb->flags = ((t->srp.is_inner_ring ? SRP_TOPOLOGY_MAC_BINDING_FLAG_IS_INNER_RING : 0) | (/* is wrapped FIXME */ 0)); clib_memcpy (mb->address, hi->hw_address, vec_len (hi->hw_address)); t->control.checksum = ~ip_csum_fold (ip_incremental_checksum (0, &t->control, vec_len (*contents) - STRUCT_OFFSET_OF (srp_generic_control_header_t, control))); { vlib_frame_t * f = vlib_get_frame_to_node (vm, hi->output_node_index); vlib_buffer_t * b; u32 * to_next = vlib_frame_vector_args (f); u32 bi; bi = vlib_buffer_add_data (vm, VLIB_BUFFER_DEFAULT_FREE_LIST_INDEX, /* buffer to append to */ ~0, *contents, vec_len (*contents)); b = vlib_get_buffer (vm, bi); vnet_buffer (b)->sw_if_index[VLIB_RX] = vnet_buffer (b)->sw_if_index[VLIB_TX] = sw_if_index; to_next[0] = bi; f->n_vectors = 1; vlib_put_frame_to_node (vm, hi->output_node_index, f); } return SRP_ERROR_CONTROL_PACKETS_PROCESSED; } typedef uword (srp_control_handler_function_t) (vlib_main_t * vm, u32 sw_if_index, u8 ** contents); static uword srp_control_input (vlib_main_t * vm, vlib_node_runtime_t * node, vlib_frame_t * from_frame) { u32 n_left_from, next_index, * from, * to_next; vlib_node_runtime_t * error_node; static u8 * contents; error_node = vlib_node_get_runtime (vm, srp_input_node.index); from = vlib_frame_vector_args (from_frame); n_left_from = from_frame->n_vectors; if (node->flags & VLIB_NODE_FLAG_TRACE) vlib_trace_frame_buffers_only (vm, node, from, n_left_from, sizeof (from[0]), sizeof (srp_input_trace_t)); next_index = node->cached_next_index; while (n_left_from > 0) { u32 n_left_to_next; vlib_get_next_frame (vm, node, next_index, to_next, n_left_to_next); while (n_left_from > 0 && n_left_to_next > 0) { u32 bi0, l2_len0, l3_len0; vlib_buffer_t * b0; u8 next0, error0; srp_generic_control_header_t * s0; bi0 = from[0]; to_next[0] = bi0; from += 1; to_next += 1; n_left_to_next -= 1; n_left_from -= 1; b0 = vlib_get_buffer (vm, bi0); s0 = (void *) (b0->data + b0->current_data); l2_len0 = vlib_buffer_length_in_chain (vm, b0); l3_len0 = l2_len0 - STRUCT_OFFSET_OF (srp_generic_control_header_t, control); error0 = SRP_ERROR_CONTROL_PACKETS_PROCESSED; error0 = s0->control.version != 0 ? SRP_ERROR_CONTROL_VERSION_NON_ZERO : error0; { u16 save0 = s0->control.checksum; u16 computed0; s0->control.checksum = 0; computed0 = ~ip_csum_fold (ip_incremental_checksum (0, &s0->control, l3_len0)); error0 = save0 != computed0 ? SRP_ERROR_CONTROL_BAD_CHECKSUM : error0; } if (error0 == SRP_ERROR_CONTROL_PACKETS_PROCESSED) { static srp_control_handler_function_t * t[SRP_N_CONTROL_PACKET_TYPE] = { [SRP_CONTROL_PACKET_TYPE_topology] = srp_topology_packet, }; srp_control_handler_function_t * f; f = 0; if (s0->control.type < ARRAY_LEN (t)) f = t[s0->control.type]; if (f) { vec_validate (contents, l2_len0 - 1); vlib_buffer_contents (vm, bi0, contents); error0 = f (vm, vnet_buffer (b0)->sw_if_index[VLIB_RX], &contents); } else error0 = SRP_ERROR_UNKNOWN_CONTROL; } b0->error = error_node->errors[error0]; next0 = 0; vlib_validate_buffer_enqueue_x1 (vm, node, next_index, to_next, n_left_to_next, bi0, next0); } vlib_put_next_frame (vm, node, next_index, n_left_to_next); } return from_frame->n_vectors; } static vlib_node_registration_t srp_control_input_node = { .function = srp_control_input, .name = "srp-control", /* Takes a vector of packets. */ .vector_size = sizeof (u32), .n_next_nodes = 1, .next_nodes = { [0] = "error-drop", }, .format_buffer = format_srp_header_with_length, .format_trace = format_srp_input_trace, .unformat_buffer = unformat_srp_header, }; static u8 * format_srp_ips_request_type (u8 * s, va_list * args) { u32 x = va_arg (*args, u32); char * t = 0; switch (x) { #define _(f,n) case SRP_IPS_REQUEST_##f: t = #f; break; foreach_srp_ips_request_type #undef _ default: return format (s, "unknown 0x%x", x); } return format (s, "%U", format_c_identifier, t); } static u8 * format_srp_ips_status (u8 * s, va_list * args) { u32 x = va_arg (*args, u32); char * t = 0; switch (x) { #define _(f,n) case SRP_IPS_STATUS_##f: t = #f; break; foreach_srp_ips_status #undef _ default: return format (s, "unknown 0x%x", x); } return format (s, "%U", format_c_identifier, t); } static u8 * format_srp_ips_state (u8 * s, va_list * args) { u32 x = va_arg (*args, u32); char * t = 0; switch (x) { #define _(f) case SRP_IPS_STATE_##f: t = #f; break; foreach_srp_ips_state #undef _ default: return format (s, "unknown 0x%x", x); } return format (s, "%U", format_c_identifier, t); } static u8 * format_srp_ring (u8 * s, va_list * args) { u32 ring = va_arg (*args, u32); return format (s, "%s", ring == SRP_RING_INNER ? "inner" : "outer"); } static u8 * format_srp_ips_header (u8 * s, va_list * args) { srp_ips_header_t * h = va_arg (*args, srp_ips_header_t *); s = format (s, "%U, %U, %U, %s-path", format_srp_ips_request_type, h->request_type, format_ethernet_address, h->originator_address, format_srp_ips_status, h->status, h->is_long_path ? "long" : "short"); return s; } static u8 * format_srp_interface (u8 * s, va_list * args) { srp_interface_t * si = va_arg (*args, srp_interface_t *); srp_interface_ring_t * ir; s = format (s, "address %U, IPS state %U", format_ethernet_address, si->my_address, format_srp_ips_state, si->current_ips_state); for (ir = si->rings; ir < si->rings + SRP_N_RING; ir++) if (ir->rx_neighbor_address_valid) s = format (s, ", %U neighbor %U", format_srp_ring, ir->ring, format_ethernet_address, ir->rx_neighbor_address); return s; } u8 * format_srp_device (u8 * s, va_list * args) { u32 hw_if_index = va_arg (*args, u32); CLIB_UNUSED (int verbose) = va_arg (*args, int); vnet_main_t * vnm = vnet_get_main(); srp_main_t * sm = &srp_main; vnet_hw_interface_t * hi = vnet_get_hw_interface (vnm, hw_if_index); srp_interface_t * si = pool_elt_at_index (sm->interface_pool, hi->hw_instance); return format (s, "%U", format_srp_interface, si); } always_inline srp_interface_t * srp_get_interface (u32 sw_if_index, srp_ring_type_t * ring) { vnet_main_t * vnm = vnet_get_main(); srp_main_t * sm = &srp_main; vnet_hw_interface_t * hi = vnet_get_sup_hw_interface (vnm, sw_if_index); srp_interface_t * si; ASSERT (hi->hw_class_index == srp_hw_interface_class.index); si = pool_elt_at_index (sm->interface_pool, hi->hw_instance); ASSERT (si->rings[SRP_RING_INNER].hw_if_index == hi->hw_if_index || si->rings[SRP_RING_OUTER].hw_if_index == hi->hw_if_index); if (ring) *ring = (hi->hw_if_index == si->rings[SRP_RING_INNER].hw_if_index ? SRP_RING_INNER : SRP_RING_OUTER); return si; } static void init_ips_packet (srp_interface_t * si, srp_ring_type_t tx_ring, srp_ips_header_t * i) { memset (i, 0, sizeof (i[0])); i->srp.ttl = 1; i->srp.is_inner_ring = tx_ring; i->srp.priority = 7; i->srp.mode = SRP_MODE_control_locally_buffered_for_host; srp_header_compute_parity (&i->srp); clib_memcpy (&i->ethernet.src_address, &si->my_address, sizeof (si->my_address)); i->ethernet.type = clib_host_to_net_u16 (ETHERNET_TYPE_SRP_CONTROL); /* Checksum will be filled in later. */ i->control.version = 0; i->control.type = SRP_CONTROL_PACKET_TYPE_ips; i->control.ttl = 255; clib_memcpy (&i->originator_address, &si->my_address, sizeof (si->my_address)); } static void tx_ips_packet (srp_interface_t * si, srp_ring_type_t tx_ring, srp_ips_header_t * i) { srp_main_t * sm = &srp_main; vnet_main_t * vnm = vnet_get_main(); vlib_main_t * vm = sm->vlib_main; vnet_hw_interface_t * hi = vnet_get_hw_interface (vnm, si->rings[tx_ring].hw_if_index); vlib_frame_t * f; vlib_buffer_t * b; u32 * to_next, bi; if (! vnet_sw_interface_is_admin_up (vnm, hi->sw_if_index)) return; if (hi->hw_class_index != srp_hw_interface_class.index) return; i->control.checksum = ~ip_csum_fold (ip_incremental_checksum (0, &i->control, sizeof (i[0]) - STRUCT_OFFSET_OF (srp_ips_header_t, control))); bi = vlib_buffer_add_data (vm, VLIB_BUFFER_DEFAULT_FREE_LIST_INDEX, /* buffer to append to */ ~0, i, sizeof (i[0])); /* FIXME trace. */ if (0) clib_warning ("%U %U", format_vnet_sw_if_index_name, vnm, hi->sw_if_index, format_srp_ips_header, i); b = vlib_get_buffer (vm, bi); vnet_buffer (b)->sw_if_index[VLIB_RX] = vnet_buffer (b)->sw_if_index[VLIB_TX] = hi->sw_if_index; f = vlib_get_frame_to_node (vm, hi->output_node_index); to_next = vlib_frame_vector_args (f); to_next[0] = bi; f->n_vectors = 1; vlib_put_frame_to_node (vm, hi->output_node_index, f); } static int requests_switch (srp_ips_request_type_t r) { static u8 t[16] = { [SRP_IPS_REQUEST_forced_switch] = 1, [SRP_IPS_REQUEST_manual_switch] = 1, [SRP_IPS_REQUEST_signal_fail] = 1, [SRP_IPS_REQUEST_signal_degrade] = 1, }; return (int) r < ARRAY_LEN (t) ? t[r] : 0; } /* Called when an IPS control packet is received on given interface. */ void srp_ips_rx_packet (u32 sw_if_index, srp_ips_header_t * h) { vnet_main_t * vnm = vnet_get_main(); vlib_main_t * vm = srp_main.vlib_main; srp_ring_type_t rx_ring; srp_interface_t * si = srp_get_interface (sw_if_index, &rx_ring); srp_interface_ring_t * ir = &si->rings[rx_ring]; /* FIXME trace. */ if (0) clib_warning ("%U %U %U", format_time_interval, "h:m:s:u", vlib_time_now (vm), format_vnet_sw_if_index_name, vnm, sw_if_index, format_srp_ips_header, h); /* Ignore self-generated IPS packets. */ if (! memcmp (h->originator_address, si->my_address, sizeof (h->originator_address))) goto done; /* Learn neighbor address from short path messages. */ if (! h->is_long_path) { if (ir->rx_neighbor_address_valid && memcmp (ir->rx_neighbor_address, h->originator_address, sizeof (ir->rx_neighbor_address))) { ASSERT (0); } ir->rx_neighbor_address_valid = 1; clib_memcpy (ir->rx_neighbor_address, h->originator_address, sizeof (ir->rx_neighbor_address)); } switch (si->current_ips_state) { case SRP_IPS_STATE_idle: /* Received {REQ,NEIGHBOR,W,S} in idle state: wrap. */ if (requests_switch (h->request_type) && ! h->is_long_path && h->status == SRP_IPS_STATUS_wrapped) { srp_ips_header_t to_tx[2]; si->current_ips_state = SRP_IPS_STATE_wrapped; si->hw_wrap_function (si->rings[SRP_SIDE_A].hw_if_index, /* enable_wrap */ 1); si->hw_wrap_function (si->rings[SRP_SIDE_B].hw_if_index, /* enable_wrap */ 1); init_ips_packet (si, rx_ring ^ 0, &to_tx[0]); to_tx[0].request_type = SRP_IPS_REQUEST_idle; to_tx[0].status = SRP_IPS_STATUS_wrapped; to_tx[0].is_long_path = 0; tx_ips_packet (si, rx_ring ^ 0, &to_tx[0]); init_ips_packet (si, rx_ring ^ 1, &to_tx[1]); to_tx[1].request_type = h->request_type; to_tx[1].status = SRP_IPS_STATUS_wrapped; to_tx[1].is_long_path = 1; tx_ips_packet (si, rx_ring ^ 1, &to_tx[1]); } break; case SRP_IPS_STATE_wrapped: if (! h->is_long_path && h->request_type == SRP_IPS_REQUEST_idle && h->status == SRP_IPS_STATUS_idle) { si->current_ips_state = SRP_IPS_STATE_idle; si->hw_wrap_function (si->rings[SRP_SIDE_A].hw_if_index, /* enable_wrap */ 0); si->hw_wrap_function (si->rings[SRP_SIDE_B].hw_if_index, /* enable_wrap */ 0); } break; case SRP_IPS_STATE_pass_thru: /* FIXME */ break; default: abort (); break; } done: ; } /* Preform local IPS request on given interface. */ void srp_ips_local_request (u32 sw_if_index, srp_ips_request_type_t request) { vnet_main_t * vnm = vnet_get_main(); srp_main_t * sm = &srp_main; srp_ring_type_t rx_ring; srp_interface_t * si = srp_get_interface (sw_if_index, &rx_ring); srp_interface_ring_t * ir = &si->rings[rx_ring]; if (request == SRP_IPS_REQUEST_wait_to_restore) { if (si->current_ips_state != SRP_IPS_STATE_wrapped) return; if (! ir->waiting_to_restore) { ir->wait_to_restore_start_time = vlib_time_now (sm->vlib_main); ir->waiting_to_restore = 1; } } else { /* FIXME handle local signal fail. */ ir->wait_to_restore_start_time = 0; ir->waiting_to_restore = 0; } /* FIXME trace. */ if (0) clib_warning ("%U %U", format_vnet_sw_if_index_name, vnm, sw_if_index, format_srp_ips_request_type, request); } static void maybe_send_ips_message (srp_interface_t * si) { srp_main_t * sm = &srp_main; srp_ips_header_t to_tx[2]; srp_ring_type_t rx_ring = SRP_RING_OUTER; srp_interface_ring_t * r0 = &si->rings[rx_ring ^ 0]; srp_interface_ring_t * r1 = &si->rings[rx_ring ^ 1]; f64 now = vlib_time_now (sm->vlib_main); if (! si->ips_process_enable) return; if (si->current_ips_state == SRP_IPS_STATE_wrapped && r0->waiting_to_restore && r1->waiting_to_restore && now >= r0->wait_to_restore_start_time + si->config.wait_to_restore_idle_delay && now >= r1->wait_to_restore_start_time + si->config.wait_to_restore_idle_delay) { si->current_ips_state = SRP_IPS_STATE_idle; r0->waiting_to_restore = r1->waiting_to_restore = 0; r0->wait_to_restore_start_time = r1->wait_to_restore_start_time = 0; } if (si->current_ips_state != SRP_IPS_STATE_idle) return; init_ips_packet (si, rx_ring ^ 0, &to_tx[0]); init_ips_packet (si, rx_ring ^ 1, &to_tx[1]); if (si->current_ips_state == SRP_IPS_STATE_idle) { to_tx[0].request_type = to_tx[1].request_type = SRP_IPS_REQUEST_idle; to_tx[0].status = to_tx[1].status = SRP_IPS_STATUS_idle; to_tx[0].is_long_path = to_tx[1].is_long_path = 0; } else if (si->current_ips_state == SRP_IPS_STATE_wrapped) { to_tx[0].request_type = (si->rings[rx_ring ^ 0].waiting_to_restore ? SRP_IPS_REQUEST_wait_to_restore : SRP_IPS_REQUEST_signal_fail); to_tx[1].request_type = (si->rings[rx_ring ^ 1].waiting_to_restore ? SRP_IPS_REQUEST_wait_to_restore : SRP_IPS_REQUEST_signal_fail); to_tx[0].status = to_tx[1].status = SRP_IPS_STATUS_wrapped; to_tx[0].is_long_path = 0; to_tx[1].is_long_path = 1; } tx_ips_packet (si, rx_ring ^ 0, &to_tx[0]); tx_ips_packet (si, rx_ring ^ 1, &to_tx[1]); } static uword srp_ips_process (vlib_main_t * vm, vlib_node_runtime_t * rt, vlib_frame_t * f) { srp_main_t * sm = &srp_main; srp_interface_t * si; while (1) { pool_foreach (si, sm->interface_pool, ({ maybe_send_ips_message (si); })); vlib_process_suspend (vm, 1.0); } return 0; } vlib_node_registration_t srp_ips_process_node = { .function = srp_ips_process, .type = VLIB_NODE_TYPE_PROCESS, .name = "srp-ips-process", .state = VLIB_NODE_STATE_DISABLED, }; static clib_error_t * srp_init (vlib_main_t * vm) { srp_main_t * sm = &srp_main; sm->default_data_ttl = 255; sm->vlib_main = vm; vlib_register_node (vm, &srp_ips_process_node); vlib_register_node (vm, &srp_input_node); vlib_register_node (vm, &srp_control_input_node); srp_setup_node (vm, srp_input_node.index); return 0; } VLIB_INIT_FUNCTION (srp_init);