| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302 |
- # SPDX-FileCopyrightText: 2022-2023 Espressif Systems (Shanghai) CO LTD
- # SPDX-License-Identifier: CC0-1.0
- import contextlib
- import logging
- import os
- import socket
- from multiprocessing import Pipe, Process, connection
- from typing import Iterator
- import pytest
- from pytest_embedded_idf import IdfDut
- from scapy.all import Ether, raw
- ETH_TYPE = 0x3300
- class EthTestIntf(object):
- def __init__(self, eth_type: int, my_if: str = ''):
- self.target_if = ''
- self.eth_type = eth_type
- self.find_target_if(my_if)
- def find_target_if(self, my_if: str = '') -> None:
- # try to determine which interface to use
- netifs = os.listdir('/sys/class/net/')
- netifs.sort(reverse=True)
- logging.info('detected interfaces: %s', str(netifs))
- for netif in netifs:
- # if no interface defined, try to find it automatically
- if my_if == '':
- if netif.find('eth') == 0 or netif.find('enp') == 0 or netif.find('eno') == 0:
- self.target_if = netif
- break
- else:
- if netif.find(my_if) == 0:
- self.target_if = my_if
- break
- if self.target_if == '':
- raise RuntimeError('network interface not found')
- logging.info('Use %s for testing', self.target_if)
- @contextlib.contextmanager
- def configure_eth_if(self, eth_type:int=0) -> Iterator[socket.socket]:
- if eth_type == 0:
- eth_type = self.eth_type
- so = socket.socket(socket.AF_PACKET, socket.SOCK_RAW, socket.htons(eth_type))
- so.bind((self.target_if, 0))
- try:
- yield so
- finally:
- so.close()
- def send_eth_packet(self, mac: str) -> None:
- with self.configure_eth_if() as so:
- 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=self.eth_type) / raw(payload)
- try:
- so.send(raw(eth_frame))
- except Exception as e:
- raise e
- def recv_resp_poke(self, mac:str, i:int=0) -> None:
- eth_type_ctrl = self.eth_type + 1
- with self.configure_eth_if(eth_type_ctrl) as so:
- so.settimeout(30)
- for _ in range(10):
- try:
- eth_frame = Ether(so.recv(60))
- except Exception as e:
- raise e
- if mac == eth_frame.src and eth_frame.load[0] == 0xfa:
- if eth_frame.load[1] != i:
- raise RuntimeError('Missed Poke Packet')
- logging.info('Poke Packet received...')
- 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))
- break
- else:
- logging.warning('Unexpected Control packet')
- logging.warning('Expected Ctrl command: 0xfa, actual: 0x%x', eth_frame.load[0])
- logging.warning('Source MAC %s', eth_frame.src)
- else:
- raise RuntimeError('No Poke Packet!')
- def traffic_gen(self, mac: str, pipe_rcv:connection.Connection) -> None:
- with self.configure_eth_if() as so:
- payload = bytes.fromhex('ff') # DUMMY_TRAFFIC code
- payload += bytes(1485)
- eth_frame = Ether(dst=mac, src=so.getsockname()[4], type=self.eth_type) / raw(payload)
- try:
- while pipe_rcv.poll() is not True:
- so.send(raw(eth_frame))
- except Exception as e:
- raise e
- def ethernet_test(dut: IdfDut) -> None:
- dut.run_all_single_board_cases(group='ethernet', timeout=980)
- def ethernet_int_emac_hal_test(dut: IdfDut) -> None:
- dut.run_all_single_board_cases(group='emac_hal')
- def ethernet_l2_test(dut: IdfDut) -> None:
- target_if = EthTestIntf(ETH_TYPE)
- dut.expect_exact('Press ENTER to see the list of tests')
- dut.write('\n')
- dut.expect_exact('Enter test for running.')
- with target_if.configure_eth_if() as so:
- so.settimeout(30)
- dut.write('"ethernet broadcast transmit"')
- res = dut.expect(
- 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})'
- )
- # wait for POKE msg to be sure the switch already started forwarding the port's traffic
- # (there might be slight delay due to the RSTP execution)
- dut_mac = res.group(1).decode('utf-8')
- target_if.recv_resp_poke(mac=dut_mac)
- for _ in range(10):
- eth_frame = Ether(so.recv(1024))
- if dut_mac == eth_frame.src:
- break
- else:
- raise RuntimeError('No broadcast received from expected DUT MAC addr')
- for i in range(0, 1010):
- if eth_frame.load[i] != i & 0xff:
- raise RuntimeError('Packet content mismatch')
- dut.expect_unity_test_output()
- dut.expect_exact("Enter next test, or 'enter' to see menu")
- dut.write('"ethernet recv_pkt"')
- res = dut.expect(
- 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})'
- )
- dut_mac = res.group(1).decode('utf-8')
- # wait for POKE msg to be sure the switch already started forwarding the port's traffic
- # (there might be slight delay due to the RSTP execution)
- target_if.recv_resp_poke(mac=dut_mac)
- target_if.send_eth_packet('ff:ff:ff:ff:ff:ff') # broadcast frame
- target_if.send_eth_packet('01:00:5e:00:00:00') # IPv4 multicast frame (some SPI Eth modules filter multicast other than IP)
- target_if.send_eth_packet(mac=dut_mac) # unicast frame
- dut.expect_unity_test_output(extra_before=res.group(1))
- dut.expect_exact("Enter next test, or 'enter' to see menu")
- dut.write('"ethernet start/stop stress test under heavy traffic"')
- res = dut.expect(
- 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})'
- )
- dut_mac = res.group(1).decode('utf-8')
- # Start/stop under heavy Tx traffic
- for tx_i in range(10):
- target_if.recv_resp_poke(dut_mac, tx_i)
- dut.expect_exact('Ethernet stopped')
- for rx_i in range(10):
- target_if.recv_resp_poke(dut_mac, rx_i)
- # Start/stop under heavy Rx traffic
- pipe_rcv, pipe_send = Pipe(False)
- tx_proc = Process(target=target_if.traffic_gen, args=(dut_mac, pipe_rcv, ))
- tx_proc.start()
- dut.expect_exact('Ethernet stopped')
- pipe_send.send(0) # just send some dummy data
- tx_proc.join(5)
- if tx_proc.exitcode is None:
- tx_proc.terminate()
- dut.expect_unity_test_output(extra_before=res.group(1))
- # ----------- IP101 -----------
- @pytest.mark.esp32
- @pytest.mark.ethernet
- @pytest.mark.parametrize('config', [
- 'default_ip101',
- 'release_ip101',
- 'single_core_ip101'
- ], indirect=True)
- @pytest.mark.flaky(reruns=3, reruns_delay=5)
- def test_esp_ethernet(dut: IdfDut) -> None:
- ethernet_test(dut)
- @pytest.mark.esp32
- @pytest.mark.ethernet
- @pytest.mark.parametrize('config', [
- 'default_ip101',
- ], indirect=True)
- def test_esp_emac_hal(dut: IdfDut) -> None:
- ethernet_int_emac_hal_test(dut)
- @pytest.mark.esp32
- @pytest.mark.ip101
- @pytest.mark.parametrize('config', [
- 'default_ip101',
- ], indirect=True)
- def test_esp_eth_ip101(dut: IdfDut) -> None:
- ethernet_l2_test(dut)
- # ----------- LAN8720 -----------
- @pytest.mark.esp32
- @pytest.mark.eth_lan8720
- @pytest.mark.nightly_run
- @pytest.mark.parametrize('config', [
- 'default_lan8720',
- ], indirect=True)
- def test_esp_eth_lan8720(dut: IdfDut) -> None:
- ethernet_test(dut)
- dut.serial.hard_reset()
- ethernet_l2_test(dut)
- # ----------- RTL8201 -----------
- @pytest.mark.esp32
- @pytest.mark.eth_rtl8201
- @pytest.mark.nightly_run
- @pytest.mark.parametrize('config', [
- 'default_rtl8201',
- ], indirect=True)
- def test_esp_eth_rtl8201(dut: IdfDut) -> None:
- ethernet_test(dut)
- dut.serial.hard_reset()
- ethernet_l2_test(dut)
- # ----------- KSZ8041 -----------
- @pytest.mark.esp32
- @pytest.mark.eth_ksz8041
- @pytest.mark.nightly_run
- @pytest.mark.parametrize('config', [
- 'default_ksz8041',
- ], indirect=True)
- def test_esp_eth_ksz8041(dut: IdfDut) -> None:
- ethernet_test(dut)
- dut.serial.hard_reset()
- ethernet_l2_test(dut)
- # ----------- DP83848 -----------
- @pytest.mark.esp32
- @pytest.mark.eth_dp83848
- @pytest.mark.nightly_run
- @pytest.mark.parametrize('config', [
- 'default_dp83848',
- ], indirect=True)
- def test_esp_eth_dp83848(dut: IdfDut) -> None:
- ethernet_test(dut)
- dut.serial.hard_reset()
- ethernet_l2_test(dut)
- # ----------- W5500 -----------
- @pytest.mark.esp32
- @pytest.mark.eth_w5500
- @pytest.mark.nightly_run
- @pytest.mark.parametrize('config', [
- 'default_w5500',
- ], indirect=True)
- def test_esp_eth_w5500(dut: IdfDut) -> None:
- ethernet_test(dut)
- dut.serial.hard_reset()
- ethernet_l2_test(dut)
- # ----------- KSZ8851SNL -----------
- @pytest.mark.esp32
- @pytest.mark.eth_ksz8851snl
- @pytest.mark.nightly_run
- @pytest.mark.parametrize('config', [
- 'default_ksz8851snl',
- ], indirect=True)
- def test_esp_eth_ksz8851snl(dut: IdfDut) -> None:
- ethernet_test(dut)
- dut.serial.hard_reset()
- ethernet_l2_test(dut)
- # ----------- DM9051 -----------
- @pytest.mark.esp32
- @pytest.mark.eth_dm9051
- @pytest.mark.nightly_run
- @pytest.mark.parametrize('config', [
- 'default_dm9051',
- ], indirect=True)
- def test_esp_eth_dm9051(dut: IdfDut) -> None:
- ethernet_test(dut)
- dut.serial.hard_reset()
- ethernet_l2_test(dut)
|