| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160 |
- import os
- import re
- import socket
- from collections.abc import Callable
- from threading import Thread
- import tiny_test_fw
- import ttfw_idf
- from scapy.all import Ether, raw
- from ttfw_idf import TestFormat
- try:
- import typing # noqa: F401 # pylint: disable=unused-import
- except ImportError:
- pass
- def configure_eth_if(func): # type: (typing.Any) -> typing.Any
- def inner(*args, **kwargs): # type: (typing.Any, typing.Any) -> typing.Any
- # try to determine which interface to use
- netifs = os.listdir('/sys/class/net/')
- target_if = ''
- print('detected interfaces: ' + str(netifs))
- for netif in netifs:
- if netif.find('eth') == 0 or netif.find('enp') == 0 or netif.find('eno') == 0:
- target_if = netif
- break
- if target_if == '':
- raise Exception('no network interface found')
- print('Use ' + target_if + ' for testing')
- so = socket.socket(socket.AF_PACKET, socket.SOCK_RAW, 0x2222)
- so.bind((target_if, 0))
- func(so, *args, **kwargs)
- so.close()
- return inner
- @configure_eth_if
- def check_eth_recv_packet(so): # type: (socket.socket) -> None
- so.settimeout(10)
- try:
- eth_frame = Ether(so.recv(1024))
- for i in range(0, 1010):
- if eth_frame.load[i] != i & 0xff:
- raise Exception('Packet content mismatch')
- except Exception as e:
- raise e
- @configure_eth_if
- def send_eth_packet(so, mac): # type: (socket.socket, str) -> None
- so.settimeout(10)
- payload = bytearray(1010)
- for i, _ in enumerate(payload):
- payload[i] = i & 0xff
- eth_frame = Ether(dst=mac, src=so.getsockname()[4], type=0x2222) / raw(payload)
- try:
- so.send(raw(eth_frame))
- except Exception as e:
- raise e
- @configure_eth_if
- def recv_resp_poke(so, i): # type: (socket.socket, int) -> None
- so.settimeout(10)
- try:
- eth_frame = Ether(so.recv(60))
- if eth_frame.type == 0x2222 and eth_frame.load[0] == 0xfa:
- if eth_frame.load[1] != i:
- raise Exception('Missed Poke Packet')
- eth_frame.dst = eth_frame.src
- eth_frame.src = so.getsockname()[4]
- eth_frame.load = bytes.fromhex('fb') # POKE_RESP code
- so.send(raw(eth_frame))
- except Exception as e:
- raise e
- @configure_eth_if
- def traffic_gen(so, mac, enabled): # type: (socket.socket, str, Callable) -> None
- payload = bytes.fromhex('ff') # DUMMY_TRAFFIC code
- payload += bytes(1485)
- eth_frame = Ether(dst=mac, src=so.getsockname()[4], type=0x2222) / raw(payload)
- try:
- while enabled() == 1:
- so.send(raw(eth_frame))
- except Exception as e:
- raise e
- def test_component_ut_esp_eth(env, appname): # type: (tiny_test_fw.Env, str) -> None
- dut = env.get_dut('esp_eth', 'components/esp_eth/test_apps', app_config_name=appname)
- dut.start_app()
- stdout = dut.expect('Press ENTER to see the list of tests', full_stdout=True)
- dut.write('"start_and_stop"')
- stdout += dut.expect("Enter next test, or 'enter' to see menu", full_stdout=True)
- ttfw_idf.ComponentUTResult.parse_result(stdout, test_format=TestFormat.UNITY_BASIC)
- dut.write('"get_set_mac"')
- stdout = dut.expect("Enter next test, or 'enter' to see menu", full_stdout=True)
- ttfw_idf.ComponentUTResult.parse_result(stdout, test_format=TestFormat.UNITY_BASIC)
- dut.write('"ethernet_broadcast_transmit"')
- check_eth_recv_packet()
- stdout = dut.expect("Enter next test, or 'enter' to see menu", full_stdout=True)
- ttfw_idf.ComponentUTResult.parse_result(stdout, test_format=TestFormat.UNITY_BASIC)
- dut.write('"recv_pkt"')
- expect_result = dut.expect(re.compile(
- r'([\s\S]*)'
- r'DUT MAC: ([0-9A-Fa-f]{2}:[0-9A-Fa-f]{2}:[0-9A-Fa-f]{2}:[0-9A-Fa-f]{2}:[0-9A-Fa-f]{2}:[0-9A-Fa-f]{2})'),
- timeout=10
- )
- stdout = expect_result[0]
- send_eth_packet('ff:ff:ff:ff:ff:ff') # broadcast frame
- send_eth_packet('01:00:00:00:00:00') # multicast frame
- send_eth_packet(expect_result[1]) # unicast frame
- stdout += dut.expect("Enter next test, or 'enter' to see menu", full_stdout=True)
- ttfw_idf.ComponentUTResult.parse_result(stdout, test_format=TestFormat.UNITY_BASIC)
- dut.write('"start_stop_stress_test"')
- expect_result = dut.expect(re.compile(
- r'([\s\S]*)'
- r'DUT MAC: ([0-9A-Fa-f]{2}:[0-9A-Fa-f]{2}:[0-9A-Fa-f]{2}:[0-9A-Fa-f]{2}:[0-9A-Fa-f]{2}:[0-9A-Fa-f]{2})'),
- timeout=10
- )
- # Start/stop under heavy Tx traffic
- for tx_i in range(10):
- recv_resp_poke(tx_i)
- # Start/stop under heavy Rx traffic
- traffic_en = 1
- thread = Thread(target=traffic_gen, args=(expect_result[1], lambda:traffic_en, ))
- thread.start()
- try:
- for rx_i in range(10):
- recv_resp_poke(rx_i)
- finally:
- traffic_en = 0
- thread.join()
- stdout = dut.expect("Enter next test, or 'enter' to see menu", full_stdout=True)
- ttfw_idf.ComponentUTResult.parse_result(stdout, test_format=TestFormat.UNITY_BASIC)
- @ttfw_idf.idf_component_unit_test(env_tag='COMPONENT_UT_IP101', target=['esp32'])
- def test_component_ut_esp_eth_ip101(env, _): # type: (tiny_test_fw.Env, typing.Any) -> None
- test_component_ut_esp_eth(env, 'ip101')
- @ttfw_idf.idf_component_unit_test(env_tag='COMPONENT_UT_LAN8720', target=['esp32'])
- def test_component_ut_esp_eth_lan8720(env, _): # type: (tiny_test_fw.Env, typing.Any) -> None
- test_component_ut_esp_eth(env, 'lan8720')
- if __name__ == '__main__':
- test_component_ut_esp_eth_ip101()
- test_component_ut_esp_eth_lan8720()
|