forked from KolibriOS/kolibrios
Magomed Kostoev (mkostoevr)
79aa4c134d
git-svn-id: svn://kolibrios.org@9395 a494cfbc-eb01-0410-851d-a64ba20cac60
348 lines
10 KiB
PHP
348 lines
10 KiB
PHP
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
;; ;;
|
|
;; Copyright (C) KolibriOS team 2004-2021. All rights reserved. ;;
|
|
;; Distributed under terms of the GNU General Public License ;;
|
|
;; ;;
|
|
;; ETHERNET.INC ;;
|
|
;; ;;
|
|
;; Ethernet network layer for KolibriOS ;;
|
|
;; ;;
|
|
;; Written by hidnplayr@kolibrios.org ;;
|
|
;; ;;
|
|
;; GNU GENERAL PUBLIC LICENSE ;;
|
|
;; Version 2, June 1991 ;;
|
|
;; ;;
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
$Revision$
|
|
|
|
ETH_FRAME_MINIMUM = 60
|
|
ETH_QUEUE_SIZE = 255
|
|
|
|
struct ETH_header
|
|
|
|
DstMAC dp ? ; destination MAC-address
|
|
SrcMAC dp ? ; source MAC-address
|
|
Type dw ? ; type of the upper-layer protocol
|
|
|
|
ends
|
|
|
|
struct ETH_DEVICE NET_DEVICE
|
|
|
|
mac dp ?
|
|
|
|
ends
|
|
|
|
iglobal
|
|
align 4
|
|
|
|
ETH_BROADCAST dp 0xffffffffffff
|
|
|
|
ETH_frame_queued dd 0 ; Number of queued frames
|
|
|
|
ETH_frame_head dd ETH_frame_head ; Pointer to next frame in the linked list
|
|
ETH_frame_tail dd ETH_frame_head ; Pointer to last frame in the linked list
|
|
|
|
endg
|
|
|
|
uglobal
|
|
align 4
|
|
ETH_input_event dd ?
|
|
endg
|
|
|
|
macro eth_init {
|
|
|
|
movi ebx, 1
|
|
mov ecx, eth_process_input
|
|
call new_sys_threads
|
|
test eax, eax
|
|
jns @f
|
|
DEBUGF DEBUG_NETWORK_ERROR,'K : cannot create kernel thread for ethernet, error %d\n', eax
|
|
@@:
|
|
|
|
}
|
|
|
|
align 4
|
|
; This function is called by ethernet drivers.
|
|
; Push the received ethernet packet onto the ethernet input queue.
|
|
;
|
|
; Input:
|
|
; [esp] = Pointer to buffer
|
|
; [esp + 4] = Return address (yes, really)
|
|
;
|
|
; Example:
|
|
; push .retaddr
|
|
; push buf_addr
|
|
; jmp eth_input
|
|
eth_input:
|
|
|
|
pop eax
|
|
|
|
if defined NETWORK_SANITY_CHECKS
|
|
cmp eax, [net_buffs_low]
|
|
jb .assert_mbuff
|
|
cmp eax, [net_buffs_high]
|
|
ja .assert_mbuff
|
|
test eax, 0x7ff
|
|
jnz .assert_mbuff
|
|
end if
|
|
|
|
spin_lock_irqsave
|
|
|
|
cmp [ETH_frame_queued], ETH_QUEUE_SIZE
|
|
jae .full
|
|
inc [ETH_frame_queued]
|
|
|
|
; Add frame to the end of the linked list
|
|
mov [eax + NET_BUFF.NextPtr], ETH_frame_head
|
|
|
|
mov ebx, [ETH_frame_tail]
|
|
mov [eax + NET_BUFF.PrevPtr], ebx
|
|
|
|
mov [ETH_frame_tail], eax
|
|
mov [ebx + NET_BUFF.NextPtr], eax
|
|
|
|
spin_unlock_irqrestore
|
|
|
|
; Mark it as being an Ethernet Frame
|
|
mov [eax + NET_BUFF.type], NET_BUFF_ETH
|
|
|
|
; Now queue an event to process it
|
|
xor edx, edx
|
|
mov eax, [ETH_input_event]
|
|
mov ebx, [eax + EVENT.id]
|
|
xor esi, esi
|
|
call raise_event
|
|
|
|
ret
|
|
|
|
.full:
|
|
mov ebx, [eax + NET_BUFF.device]
|
|
inc [ebx + NET_DEVICE.packets_rx_ovr]
|
|
DEBUGF DEBUG_NETWORK_VERBOSE, "ETH incoming queue is full, discarding packet!\n"
|
|
spin_unlock_irqrestore
|
|
stdcall net_buff_free, eax
|
|
ret
|
|
|
|
if defined NETWORK_SANITY_CHECKS
|
|
.assert_mbuff:
|
|
DEBUGF DEBUG_NETWORK_ERROR, "eth_input: invalid buffer 0x%x\n", eax
|
|
DEBUGF DEBUG_NETWORK_ERROR, "eth_input: caller=0x%x\n", [esp+4]
|
|
xor eax, eax
|
|
ret
|
|
end if
|
|
|
|
|
|
|
|
;-----------------------------------------------------------------;
|
|
; ;
|
|
; eth_process_input: Process packets from ethernet input queue. ;
|
|
; ;
|
|
; IN: / ;
|
|
; ;
|
|
; OUT: / ;
|
|
; ;
|
|
;-----------------------------------------------------------------;
|
|
align 4
|
|
eth_process_input:
|
|
|
|
xor esi, esi
|
|
mov ecx, MANUAL_DESTROY
|
|
call create_event
|
|
mov [ETH_input_event], eax
|
|
pushf
|
|
.wait:
|
|
popf
|
|
mov eax, [ETH_input_event]
|
|
mov ebx, [eax + EVENT.id]
|
|
call wait_event
|
|
|
|
.loop:
|
|
pushf
|
|
cli
|
|
cmp [ETH_frame_queued], 0
|
|
je .wait
|
|
|
|
dec [ETH_frame_queued]
|
|
|
|
mov esi, [ETH_frame_head]
|
|
mov ebx, [esi + NET_BUFF.NextPtr]
|
|
|
|
mov [ETH_frame_head], ebx
|
|
mov [ebx + NET_BUFF.PrevPtr], ETH_frame_head
|
|
|
|
popf
|
|
|
|
mov eax, [esi + NET_BUFF.offset]
|
|
add eax, esi
|
|
mov ecx, [esi + NET_BUFF.length]
|
|
mov ebx, [esi + NET_BUFF.device]
|
|
|
|
pushd .loop ; return address for protocol handler
|
|
push esi ; keep pointer to NET_BUFF on stack
|
|
|
|
DEBUGF DEBUG_NETWORK_VERBOSE, "ETH_input: size=%u\n", ecx
|
|
sub ecx, sizeof.ETH_header
|
|
jb .err
|
|
|
|
; Set registers for protocol handlers
|
|
lea edx, [eax + sizeof.ETH_header]
|
|
mov ax, [eax + ETH_header.Type]
|
|
|
|
; Place protocol handlers here
|
|
cmp ax, ETHER_PROTO_IPv4
|
|
je ipv4_input
|
|
|
|
cmp ax, ETHER_PROTO_ARP
|
|
je arp_input
|
|
|
|
; cmp ax, ETHER_PROTO_IPv6
|
|
; je ipv6_input
|
|
|
|
; cmp ax, ETHER_PROTO_PPP_DISCOVERY
|
|
; je pppoe_discovery_input
|
|
|
|
; cmp ax, ETHER_PROTO_PPP_SESSION
|
|
; je pppoe_session_input
|
|
|
|
DEBUGF DEBUG_NETWORK_VERBOSE, "ETH_input: Unknown packet type=%x\n", ax
|
|
|
|
.drop:
|
|
mov eax, [esp]
|
|
mov eax, [eax + NET_BUFF.device]
|
|
inc [eax + NET_DEVICE.packets_rx_drop]
|
|
DEBUGF DEBUG_NETWORK_VERBOSE, "ETH_input: dropping\n"
|
|
call net_buff_free
|
|
ret
|
|
|
|
.err:
|
|
mov eax, [esp]
|
|
mov eax, [eax + NET_BUFF.device]
|
|
inc [eax + NET_DEVICE.packets_rx_err]
|
|
DEBUGF DEBUG_NETWORK_VERBOSE, "ETH_input: invalid frame received\n"
|
|
call net_buff_free
|
|
ret
|
|
|
|
|
|
|
|
|
|
;-----------------------------------------------------------------;
|
|
; ;
|
|
; eth_output ;
|
|
; ;
|
|
; IN: ax = protocol ;
|
|
; ebx = device ptr ;
|
|
; ecx = payload size ;
|
|
; edx = pointer to destination mac ;
|
|
; ;
|
|
; OUT: eax = start of net frame / 0 on error ;
|
|
; ebx = device ptr ;
|
|
; ecx = payload size ;
|
|
; edi = start of payload ;
|
|
; ;
|
|
;-----------------------------------------------------------------;
|
|
align 4
|
|
eth_output:
|
|
|
|
DEBUGF DEBUG_NETWORK_VERBOSE, "ETH_output: size=%u device=%x\n", ecx, ebx
|
|
|
|
cmp ecx, [ebx + ETH_DEVICE.mtu]
|
|
ja .too_large
|
|
|
|
push ecx
|
|
push ax edx
|
|
|
|
add ecx, sizeof.ETH_header + NET_BUFF.data
|
|
stdcall net_buff_alloc, ecx
|
|
test eax, eax
|
|
jz .out_of_ram
|
|
|
|
mov [eax + NET_BUFF.type], NET_BUFF_ETH
|
|
mov [eax + NET_BUFF.device], ebx
|
|
mov [eax + NET_BUFF.offset], NET_BUFF.data
|
|
lea edi, [eax + NET_BUFF.data]
|
|
|
|
pop esi
|
|
movsd
|
|
movsw
|
|
lea esi, [ebx + ETH_DEVICE.mac]
|
|
movsd
|
|
movsw
|
|
pop ax
|
|
stosw
|
|
|
|
lea eax, [edi - sizeof.ETH_header - NET_BUFF.data] ; Set eax to buffer start
|
|
pop ecx
|
|
|
|
lea edx, [ecx + sizeof.ETH_header] ; Set edx to complete buffer size
|
|
cmp edx, ETH_FRAME_MINIMUM
|
|
jbe .adjust_size
|
|
.done:
|
|
mov [eax + NET_BUFF.length], edx
|
|
DEBUGF DEBUG_NETWORK_VERBOSE, "ETH_output: ptr=%x size=%u\n", eax, edx
|
|
ret
|
|
|
|
.adjust_size:
|
|
mov edx, ETH_FRAME_MINIMUM
|
|
test edx, edx ; clear zero flag
|
|
jmp .done
|
|
|
|
.out_of_ram:
|
|
inc [ebx + NET_DEVICE.packets_tx_drop]
|
|
DEBUGF DEBUG_NETWORK_VERBOSE, "ETH_output: Out of ram!\n"
|
|
add esp, 4+2
|
|
pop ecx
|
|
xor eax, eax
|
|
ret
|
|
|
|
.too_large:
|
|
inc [eax + NET_DEVICE.packets_tx_err]
|
|
DEBUGF DEBUG_NETWORK_VERBOSE, "ETH_output: Packet too large!\n"
|
|
xor eax, eax
|
|
ret
|
|
|
|
|
|
|
|
;-----------------------------------------------------------------;
|
|
; ;
|
|
; eth_api: Part of system function 76. ;
|
|
; ;
|
|
; IN: bl = subfunction number ;
|
|
; bh = device number ;
|
|
; ecx, edx, .. depends on subfunction ;
|
|
; ;
|
|
; OUT: depends on subfunction ;
|
|
; ;
|
|
;-----------------------------------------------------------------;
|
|
align 4
|
|
eth_api:
|
|
|
|
cmp bh, NET_DEVICES_MAX
|
|
ja .error
|
|
movzx eax, bh
|
|
mov eax, dword [net_device_list + 4*eax]
|
|
cmp [eax + NET_DEVICE.device_type], NET_DEVICE_ETH
|
|
jne .error
|
|
|
|
and ebx, 0xff
|
|
cmp ebx, .number
|
|
ja .error
|
|
jmp dword [.table + 4*ebx]
|
|
|
|
.table:
|
|
dd .read_mac ; 0
|
|
.number = ($ - .table) / 4 - 1
|
|
|
|
.error:
|
|
or eax, -1
|
|
ret
|
|
|
|
|
|
.read_mac:
|
|
movzx ebx, word [eax + ETH_DEVICE.mac]
|
|
mov eax, dword [eax + ETH_DEVICE.mac + 2]
|
|
mov [esp+20+4], ebx ; FIXME
|
|
ret
|
|
|
|
|