;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; RDOS operating system ; Copyright (C) 1988-2000, Leif Ekblad ; ; This program is free software; you can redistribute it and/or modify ; it under the terms of the GNU General Public License as published by ; the Free Software Foundation; either version 2 of the License, or ; (at your option) any later version. The only exception to this rule ; is for commercial usage in embedded systems. For information on ; usage in commercial embedded systems, contact embedded@rdos.net ; ; This program is distributed in the hope that it will be useful, ; but WITHOUT ANY WARRANTY; without even the implied warranty of ; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the ; GNU General Public License for more details. ; ; You should have received a copy of the GNU General Public License ; along with this program; if not, write to the Free Software ; Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA ; ; The author of this program may be contacted at leif@rdos.net ; ; HUB.ASM ; Implements HUB class for USB ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; include ..\os.def include ..\os.inc include ..\user.def include ..\user.inc include ..\driver.def INCLUDE ..\os\protseg.def include ..\usbdev\usb.inc include ..\usbdev\usbdev.inc include ..\usbdev\hub.inc GET_STATUS = 0 CLEAR_FEATURE = 1 SET_FEATURE = 3 GET_DESCR = 6 SET_DESCR = 7 CLEAR_TT = 8 RESET_TT = 9 GET_TT_STATE = 10 STOP_TT = 11 PORT_CONNECTION = 0 PORT_ENABLE = 1 PORT_SUSPEND = 2 PORT_OVER_CURRENT = 3 PORT_RESET = 4 PORT_POWER = 8 PORT_LOW_SPEED = 9 C_PORT_CONNECTION = 16 C_PORT_ENABLE = 17 C_PORT_SUSPEND = 18 C_PORT_OVER_CURRENT = 19 C_PORT_RESET = 20 PORT_TEST = 21 PORT_INDICATOR = 22 data SEGMENT byte public 'DATA' hub_list DW ? hub_section section_typ <> data ENDS ;;;;;;;;; INTERNAL PROCEDURES ;;;;;;;;;;; code SEGMENT byte public 'CODE' assume cs:code .386p ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ProcessHubDescr ; ; description: Process Hub descriptor ; ; Parameters: GS Hub ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ProcessHubDescr Proc near push es pushad ; mov ax,gs mov es,ax mov bx,gs:hub_control_handle ; mov di,OFFSET hub_control_data mov es:[di].usd_type,0A0h mov es:[di].usd_req,GET_DESCR mov es:[di].usd_value,2900h mov es:[di].usd_index,0 mov es:[di].usd_len,HUB_BUF_SIZE mov cx,8 WriteUsbControl ; mov cx,HUB_BUF_SIZE mov di,OFFSET hub_buf mov es:[di].uhd_type,0 ReqUsbData ; WriteUsbStatus StartUsbTransaction ; GetSystemTime add eax,1000 * 1193 adc edx,0 mov bx,gs:hub_control_wait WaitWithTimeout ; mov bx,gs:hub_control_handle WasUsbTransactionOk jc ghdDone ; mov cl,es:[di].uhd_type cmp cl,29h stc jne ghdDone ; movzx cx,es:[di].uhd_ports mov gs:hub_ports,cx ; mov cx,es:[di].uhd_info mov gs:hub_info,cx ; movzx cx,es:[di].uhd_power_time shl cx,1 mov gs:hub_power_time,cx ; mov cx,gs:hub_ports mov bx,OFFSET hub_port_arr phdLoop: mov gs:[bx].hps_status,0 mov gs:[bx].hps_req_reset,0 mov gs:[bx].hps_ext_reset,0 mov gs:[bx].hps_dev_port,0 mov gs:[bx].hps_attach_thread,0 mov gs:[bx].hps_detach_thread,0 mov gs:[bx].hps_timeout,0 mov gs:[bx].hps_timeout+4,0 mov gs:[bx].hps_power_timeout,0 mov gs:[bx].hps_power_timeout+4,0 add bx,32 loop phdLoop ; mov gs:hub_attached,1 clc ghdDone: popad pop es ret ProcessHubDescr Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SetPortFeature ; ; description: Set port feature ; ; Parameters: GS Hub ; DX Port ; AX Feature ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; SetPortFeature Proc near push es pushad ; mov bx,gs mov es,bx mov bx,gs:hub_control_handle ; mov di,OFFSET hub_control_data mov es:[di].usd_type,23h mov es:[di].usd_req,SET_FEATURE mov es:[di].usd_value,ax mov es:[di].usd_index,dx mov es:[di].usd_len,0 mov cx,8 WriteUsbControl ReqUsbStatus StartUsbTransaction ; GetSystemTime add eax,1000 * 1193 adc edx,0 mov bx,gs:hub_control_wait WaitWithTimeout ; mov bx,gs:hub_control_handle WasUsbTransactionOk ; popad pop es ret SetPortFeature Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ClearPortFeature ; ; description: Clear port feature ; ; Parameters: GS Hub ; DX Port ; AX Feature ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ClearPortFeature Proc near push es pushad ; mov bx,gs mov es,bx mov bx,gs:hub_control_handle ; mov di,OFFSET hub_control_data mov es:[di].usd_type,23h mov es:[di].usd_req,CLEAR_FEATURE mov es:[di].usd_value,ax mov es:[di].usd_index,dx mov es:[di].usd_len,0 mov cx,8 WriteUsbControl ReqUsbStatus StartUsbTransaction ; GetSystemTime add eax,1000 * 1193 adc edx,0 mov bx,gs:hub_control_wait WaitWithTimeout ; mov bx,gs:hub_control_handle WasUsbTransactionOk ; popad pop es ret ClearPortFeature Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ClearPortChange ; ; description: Clear port change ; ; Parameters: GS Hub ; DX Port ; AX Status change ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ClearPortChange Proc near test ax,1 jz cpcConnectOk ; push ax mov ax,16 call ClearPortFeature pop ax cpcConnectOk: test ax,2 jz cpcEnableOk ; push ax mov ax,17 call ClearPortFeature pop ax cpcEnableOk: test ax,4 jz cpcSuspendOk ; push ax mov ax,18 call ClearPortFeature pop ax cpcSuspendOk: test ax,8 jz cpcOverCurrentOk ; push ax mov ax,19 call ClearPortFeature pop ax cpcOverCurrentOk: test ax,10h jz cpcResetOk ; push ax mov ax,20 call ClearPortFeature pop ax cpcResetOk: ret ClearPortChange Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; NAME: GetPortStatus ; ; description: Get port status ; ; Parameters: GS Hub ; DX Port ; ; Returns: AX Port status ; DX Status change ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; GetPortStatus Proc near push es push bx push cx push di ; mov ax,gs mov es,ax mov bx,gs:hub_control_handle ; mov di,OFFSET hub_control_data mov es:[di].usd_type,0A3h mov es:[di].usd_req,GET_STATUS mov es:[di].usd_value,0 mov es:[di].usd_index,dx mov es:[di].usd_len,4 mov cx,8 WriteUsbControl ; mov cx,4 mov di,OFFSET hub_buf mov es:[di].uhd_type,0 ReqUsbData ; WriteUsbStatus StartUsbTransaction ; push edx GetSystemTime add eax,1000 * 1193 adc edx,0 mov bx,gs:hub_control_wait WaitWithTimeout pop edx ; mov bx,gs:hub_control_handle WasUsbTransactionOk jc gpsDone ; mov di,OFFSET hub_buf mov ax,es:[di+2] call ClearPortChange ; mov ax,es:[di] clc gpsDone: pop di pop cx pop bx pop es ret GetPortStatus Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CreateHub ; ; description: Create hub ; ; Parameters: GS Hub ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; CreateHub Proc near push es pushad ; CreateWait mov gs:hub_control_wait,bx ; mov bx,gs:hub_controller mov al,gs:hub_device xor dl,dl OpenUsbPipe mov gs:hub_control_handle,bx ; mov ax,gs:hub_control_handle mov bx,gs:hub_control_wait xor ecx,ecx AddWaitForUsbPipe ; mov bx,gs:hub_controller mov al,gs:hub_device mov dl,gs:hub_intr OpenUsbPipe mov gs:hub_status_handle,bx ; mov bx,gs:hub_status_handle CreateUsbReq mov gs:hub_status_req,bx ; mov cx,gs:hub_status_size xor ax,ax AddReadUsbDataReq mov gs:hub_status_sel,es ; popad pop es ret CreateHub Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: CloseHub ; ; description: Close hub ; ; Parameters: GS Hub ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; CloseHub Proc near push bx ; mov bx,gs:hub_status_req CloseUsbReq ; mov bx,gs:hub_status_handle CloseUsbPipe ; mov bx,gs:hub_control_handle CloseUsbPipe ; mov bx,gs:hub_control_wait CloseWait ; pop bx ret CloseHub Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: HubAttach ; ; description: Hub attach event ; ; Parameters: GS Hub ; DS Device sel ; BX Port array ; DX Port # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; HubAttach Proc near push ds push ax push bx ; test gs:[bx].hps_status,200h jnz haLowSpeed ; test gs:[bx].hps_status,400h jnz haHighSpeed haFullSpeed: mov ah,1 jmp haAttach haHighSpeed: mov ah,2 jmp haAttach haLowSpeed: mov ah,0 haAttach: mov al,gs:[bx].hps_dev_port LockedNotifyUsbAttach haDone: pop bx pop ax pop ds ret HubAttach Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: HubDetach ; ; description: Hub detach event ; ; Parameters: GS Hub ; DS Device sel ; DX Port array ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; HubDetach Proc near push ds push ax push bx ; mov al,gs:[bx].hps_dev_port NotifyUsbDetach ; pop bx pop ax pop ds ret HubDetach Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: InitPorts ; ; description: Init ports ; ; Parameters: GS Hub ; DS Device sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; InitPorts Proc near pushad ; mov cx,gs:hub_ports mov bx,OFFSET hub_port_arr mov si,1 ipLoop: mov dx,si mov ax,PORT_POWER call SetPortFeature ; add bx,16 inc si loop ipLoop ; popad ret InitPorts Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: AttachThread ; ; DESCRIPTION: Attach thread ; ; PARAMETERS: BX Port # ; DX Hub selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; attach_thread_name DB 'Hub Attach', 0 attach_thread: mov gs,dx mov ds,gs:hub_dev_sel ; mov si,bx dec bx shl bx,5 add bx,OFFSET hub_port_arr ; mov al,gs:[bx].hps_dev_port or al,al jnz atHasPort ; mov dx,si call fword ptr ds:allocate_hub_port_proc jc atDone ; mov gs:[bx].hps_dev_port,al atHasPort: LockUsb ; GetThread mov gs:[bx].hps_attach_thread,ax ; mov ax,gs:hub_attached or ax,ax jz atFreeUnlock ; mov gs:[bx].hps_req_reset,1 atWaitReset: mov ax,5 WaitMilliSec ; mov al,gs:[bx].hps_req_reset or al,al jnz atWaitReset ; mov cx,40 atWaitLoop: mov ax,gs:hub_attached or ax,ax jz atFreeUnlock ; mov ax,gs:[bx].hps_status test ax,1 jz atFreeUnlock ; test ax,2 jnz atIsEnabled ; mov ax,25 WaitMilliSec loop atWaitLoop ; jmp atFreeUnlock atIsEnabled: mov dx,si call HubAttach jmp atDone atFreeUnlock: mov al,gs:[bx].hps_dev_port or al,al jz atUnlock ; call fword ptr ds:free_hub_port_proc mov gs:[bx].hps_dev_port,0 atUnlock: UnlockUsb atDone: mov gs:[bx].hps_attach_thread,0 ; TerminateThread ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: DetachThread ; ; DESCRIPTION: Detach thread ; ; PARAMETERS: BX Port # ; DX Hub selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; detach_thread_name DB 'Hub Detach', 0 detach_thread: mov gs,dx mov ds,gs:hub_dev_sel ; mov si,bx dec bx shl bx,5 add bx,OFFSET hub_port_arr ; GetThread mov gs:[bx].hps_detach_thread,ax ; mov dx,si call HubDetach ; mov al,gs:[bx].hps_dev_port call fword ptr ds:free_hub_port_proc mov gs:[bx].hps_dev_port,0 dtDone: mov gs:[bx].hps_detach_thread,0 ; TerminateThread ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UpdateOnePort ; ; description: Update ports ; ; Parameters: GS Hub ; DS Device sel ; BX Port arr ; SI Port # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UpdateOnePort Proc near mov ax,gs:hub_attached or ax,ax jz uopDone ; mov al,gs:[bx].hps_ext_reset or al,al jz uopCheckReset ; mov eax,gs:[bx].hps_power_timeout or eax,gs:[bx].hps_power_timeout+4 jnz uopCheckPowerTimeout ; GetSystemTime add eax,1193 * 250 adc edx,0 mov gs:[bx].hps_power_timeout,eax mov gs:[bx].hps_power_timeout+4,edx ; mov dx,si mov ax,PORT_POWER call ClearPortFeature jmp uopNotConnected uopCheckPowerTimeout: GetSystemTime sub eax,gs:[bx].hps_power_timeout sbb edx,gs:[bx].hps_power_timeout+4 jc uopCheckReset ; mov dx,si mov ax,PORT_POWER call SetPortFeature ; mov ax,gs:hub_power_time WaitMilliSec mov gs:[bx].hps_ext_reset,0 mov gs:[bx].hps_power_timeout,0 mov gs:[bx].hps_power_timeout+4,0 uopCheckReset: mov al,gs:[bx].hps_req_reset or al,al jz uopResetOk uopDoReset: mov dx,si mov ax,PORT_RESET call SetPortFeature ; mov ax,25 WaitMilliSec uopResetDone: mov gs:[bx].hps_req_reset,0 jmp uopDone uopResetOk: mov dx,si call GetPortStatus jc uopDone ; mov gs:[bx].hps_status,ax ; test ax,100h jz uopDone ; test ax,1 jz uopNotConnected ; test ax,2 jnz uopCheckTimeout ; mov ax,gs:[bx].hps_attach_thread or ax,gs:[bx].hps_detach_thread jnz uopCheckTimeout ; mov gs:[bx].hps_attach_thread,-1 GetSystemTime add eax,1193 * 500 adc edx,0 mov gs:[bx].hps_timeout,eax mov gs:[bx].hps_timeout+4,edx ; mov dx,gs mov bx,si ; push ds push si mov ax,cs mov ds,ax mov es,ax mov edi,OFFSET attach_thread_name mov esi,OFFSET attach_thread mov ax,2 mov cx,stack0_size CreateThread pop si pop ds jmp uopDone uopNotConnected: mov al,gs:[bx].hps_dev_port or al,al jz uopDone ; mov ax,gs:[bx].hps_attach_thread or ax,gs:[bx].hps_detach_thread jnz uopCheckTimeout ; mov gs:[bx].hps_detach_thread,-1 GetSystemTime add eax,1193 * 500 adc edx,0 mov gs:[bx].hps_timeout,eax mov gs:[bx].hps_timeout+4,edx ; mov dx,gs mov bx,si ; push ds push si mov ax,cs mov ds,ax mov es,ax mov edi,OFFSET detach_thread_name mov esi,OFFSET detach_thread mov ax,2 mov cx,stack0_size CreateThread pop si pop ds jmp uopDone uopCheckTimeout: mov ax,gs:[bx].hps_attach_thread or ax,ax jz uopCheckDetach ; cmp ax,-1 jnz uopTimeoutAttach ; GetSystemTime add eax,1193 * 500 adc edx,0 mov gs:[bx].hps_timeout,eax mov gs:[bx].hps_timeout+4,edx jmp uopDone uopTimeoutAttach: GetSystemTime sub eax,gs:[bx].hps_timeout sbb edx,gs:[bx].hps_timeout+4 jc uopDone ; push bx mov bx,gs:[bx].hps_attach_thread Signal pop bx jmp uopDone uopCheckDetach: mov ax,gs:[bx].hps_detach_thread or ax,ax jz uopDone ; cmp ax,-1 jnz uopTimeoutDetach ; GetSystemTime add eax,1193 * 500 adc edx,0 mov gs:[bx].hps_timeout,eax mov gs:[bx].hps_timeout+4,edx jmp uopDone uopTimeoutDetach: GetSystemTime sub eax,gs:[bx].hps_timeout sbb edx,gs:[bx].hps_timeout+4 jc uopDone ; push bx mov bx,gs:[bx].hps_detach_thread Signal pop bx uopDone: ret UpdateOnePort Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UpdateClosedPort ; ; description: Update closed port ; ; Parameters: GS Hub ; DS Device sel ; BX Port arr ; SI Port # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UpdateClosedPort Proc near mov gs:[bx].hps_status,0 ; mov al,gs:[bx].hps_dev_port or al,al jz ucpCheckTimeout ; mov ax,gs:[bx].hps_attach_thread or ax,gs:[bx].hps_detach_thread jnz ucpCheckTimeout ; mov gs:[bx].hps_detach_thread,-1 GetSystemTime add eax,1193 * 500 adc edx,0 mov gs:[bx].hps_timeout,eax mov gs:[bx].hps_timeout+4,edx ; mov dx,gs mov bx,si ; push ds mov ax,cs mov ds,ax mov es,ax mov edi,OFFSET detach_thread_name mov esi,OFFSET detach_thread mov ax,2 mov cx,stack0_size CreateThread pop ds jmp ucpDone ucpCheckTimeout: mov ax,gs:[bx].hps_attach_thread or ax,ax jz ucpCheckDetach ; cmp ax,-1 jnz ucpTimeoutAttach ; GetSystemTime add eax,1193 * 500 adc edx,0 mov gs:[bx].hps_timeout,eax mov gs:[bx].hps_timeout+4,edx jmp ucpDone ucpTimeoutAttach: GetSystemTime sub eax,gs:[bx].hps_timeout sbb edx,gs:[bx].hps_timeout+4 jc ucpDone ; push bx mov bx,gs:[bx].hps_attach_thread Signal pop bx jmp ucpDone ucpCheckDetach: mov ax,gs:[bx].hps_detach_thread or ax,ax jz ucpDone ; cmp ax,-1 jnz ucpTimeoutDetach ; GetSystemTime add eax,1193 * 500 adc edx,0 mov gs:[bx].hps_timeout,eax mov gs:[bx].hps_timeout+4,edx jmp ucpDone ucpTimeoutDetach: GetSystemTime sub eax,gs:[bx].hps_timeout sbb edx,gs:[bx].hps_timeout+4 jc ucpDone ; push bx mov bx,gs:[bx].hps_detach_thread Signal pop bx ucpDone: ret UpdateClosedPort Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: UpdatePorts ; ; description: Update ports ; ; Parameters: GS Hub ; DS Device sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; UpdatePorts Proc near pushad ; mov cx,gs:hub_ports mov bx,OFFSET hub_port_arr mov si,1 upLoop: push bx push cx push si ; call UpdateOnePort ; pop si pop cx pop bx ; add bx,32 inc si loop upLoop ; popad ret UpdatePorts Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: hub_thread ; ; DESCRIPTION: HUB thread ; ; PARAMETERS: BX Hub selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; hub_thread_handler: xor ax,ax mov ds,ax mov es,ax ; mov gs,bx GetThread mov gs:hub_thread,ax ; call CreateHub call ProcessHubDescr jc hub_exit ; mov ds,gs:hub_dev_sel call InitPorts ; mov ax,gs:hub_power_time WaitMilliSec hub_thread_wait: mov ax,gs:hub_attached or ax,ax jz hub_exit ; mov bx,gs:hub_status_req IsUsbReqStarted jnc hub_thread_wait_signal hub_thread_restart: ClearSignal GetThread mov cx,gs:hub_status_size StartUsbReq hub_thread_wait_signal: GetSystemTime add eax,250 * 1193 adc edx,0 ; WaitForSignal WaitForSignalWithTimeout ; mov bx,gs:hub_status_req IsUsbReqReady jnc hub_thread_is_ready ; call UpdatePorts jmp hub_thread_wait_signal hub_thread_is_ready: GetUsbReqData cmp cx,gs:hub_status_size je hub_thread_handle ; call UpdatePorts jmp hub_thread_wait hub_thread_handle: mov es,gs:hub_status_sel mov bp,cx mov bx,OFFSET hub_port_arr mov si,1 xor di,di mov al,es:[di] shr al,1 mov cx,7 hub_thread_port_loop: test al,1 jz hub_thread_port_next ; push ax push bx push si push di push bp call UpdateOnePort pop bp pop di pop si pop bx pop ax hub_thread_port_next: add bx,32 inc si shr al,1 loop hub_thread_port_loop ; sub bp,1 jz hub_thread_wait ; inc di mov al,es:[di] mov cx,8 jmp hub_thread_port_loop hub_exit: mov gs:hub_attached,0 hub_wait_retry: mov cx,gs:hub_ports mov bx,OFFSET hub_port_arr mov si,1 xor dx,dx hub_wait_port_loop: movzx ax,gs:[bx].hps_dev_port or ax,gs:[bx].hps_attach_thread or ax,gs:[bx].hps_detach_thread jz hub_wait_next_port ; inc dx pusha call UpdateClosedPort popa hub_wait_next_port: add bx,32 inc si loop hub_wait_port_loop ; or dx,dx jz hub_wait_done ; mov ax,10 WaitMilliSec jmp hub_wait_retry hub_wait_done: xor ax,ax mov ds,ax mov es,ax mov fs,ax call CloseHub ; mov gs:hub_attached,-1 ; mov ax,25 WaitMilliSec ; TerminateThread ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: HexToAscii ; ; DESCRIPTION: ; ; PARAMETERS: AL Number to convert ; ; RETURNS: AX Ascii result ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; HexToAscii PROC near mov ah,al and al,0F0h rol al,1 rol al,1 rol al,1 rol al,1 cmp al,0Ah jb ok_low1 ; add al,7 ok_low1: add al,30h and ah,0Fh cmp ah,0Ah jb ok_high1 ; add ah,7 ok_high1: add ah,30h ret HexToAscii ENDP ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: usb_attach ; ; description: USB attach callback ; ; Parameters: BX Controller # ; AL Device address ; DS USB device ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; hub_name DB 'Usb Hub ', 0 usb_attach Proc far push ds push es pushad ; push ax mov eax,1000h AllocateSmallGlobalMem mov cx,SIZE usb_device_descr pop ax xor di,di push ax GetUsbDevice cmp ax,cx pop ax jne uaDone ; mov cl,es:udd_class cmp cl,9 jne uaDone ; xor dl,dl mov cx,1000h xor di,di push ax GetUsbConfig mov cx,ax pop ax or cx,cx jz uaDone ; mov dl,es:ucd_config_id xor di,di movzx cx,es:ucd_len add di,cx uaCheckLoop: mov cl,es:[di].ucd_type cmp cl,4 jne uaCheckNext ; mov cl,es:[di].uid_class cmp cl,9 je uaConfig uaCheckNext: movzx cx,es:[di].ucd_len or cx,cx jz uaDone ; add di,cx cmp di,es:ucd_size jb uaCheckLoop jmp uaDone uaConfig: ConfigUsbDevice jc uaDone ; push es push ax push ax mov eax,SIZE hub_struc AllocateSmallGlobalMem pop ax mov es:hub_controller,bx mov es:hub_device,al mov es:hub_status_size,0 mov es:hub_intr,0 mov es:hub_info,0 mov es:hub_power_time,0 mov es:hub_dev_sel,ds mov ax,es mov gs,ax pop ax pop es ; xor di,di movzx cx,es:ucd_len add di,cx uaDescrLoop: mov cl,es:[di].udd_type cmp cl,5 jne uaDescrNext uaDescrDo: mov cl,es:[di].ued_attrib and cl,3 cmp cl,3 jne uaDescrNext ; mov cl,es:[di].ued_address mov gs:hub_intr,cl ; mov cx,es:[di].ued_maxsize mov gs:hub_status_size,cx uaDescrNext: movzx cx,es:[di].ucd_len add di,cx cmp di,es:ucd_size jb uaDescrLoop uaOk: mov al,gs:hub_intr or al,al jnz uaValid ; push es mov ax,gs mov es,ax xor ax,ax mov gs,ax FreeMem pop es jmp uaDone uaValid: mov ax,SEG data mov ds,ax mov bx,gs ; EnterSection ds:hub_section mov ax,ds:hub_list or ax,ax jz uaInsEmpty uaInsLoop: mov gs,ax mov ax,gs:hub_next or ax,ax jnz uaInsLoop ; mov gs:hub_next,bx jmp uaInsDone uaInsEmpty: mov ds:hub_list,bx uaInsDone: mov gs,bx mov gs:hub_next,0 LeaveSection ds:hub_section ; push es mov eax,100h AllocateSmallGlobalMem xor di,di mov si,OFFSET hub_name uaCopyHub: mov al,cs:[si] inc si or al,al jz uaCopyDone ; stosb jmp uaCopyHub uaCopyDone: mov ax,gs:hub_controller call HexToAscii stosw ; mov al,'.' stosb ; mov al,gs:hub_device call HexToAscii stosw ; xor al,al stosb ; mov bx,gs xor di,di mov dx,cs mov ds,dx mov si,OFFSET hub_thread_handler mov ax,3 mov cx,stack0_size CreateThread ; FreeMem pop es uaDone: FreeMem ; popad pop es pop ds retf32 usb_attach Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: usb_detach ; ; description: USB detach callback ; ; Parameters: BX Controller # ; AL Device address ; DS USB device ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; usb_detach Proc far push ds push es push gs pushad ; mov dx,SEG data mov ds,dx xor dx,dx mov es,dx ; EnterSection ds:hub_section mov dx,ds:hub_list xor si,si udLoop: or dx,dx jz udNotMe ; mov gs,dx cmp bx,gs:hub_controller jne udNext ; cmp al,gs:hub_device je udFound udNext: mov si,dx mov dx,gs:hub_next jmp udLoop udFound: or si,si jz udFirst ; push es mov ax,gs:hub_next mov es,si mov es:hub_next,ax pop es jmp udClose udFirst: mov ax,gs:hub_next mov ds:hub_list,ax udClose: mov ax,gs:hub_attached cmp ax,-1 je udLeave ; mov gs:hub_attached,0 udLeave: LeaveSection ds:hub_section udSignal: mov bx,gs:hub_thread Signal ; mov ax,25 WaitMilliSec ; mov ax,gs:hub_attached cmp ax,-1 jne udSignal ; mov ax,gs mov es,ax xor ax,ax mov gs,ax FreeMem jmp udDone udNotMe: LeaveSection ds:hub_section udDone: popad pop gs pop es pop ds retf32 usb_detach Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: IsUsbHubPortConnected ; ; description: Check if Hub port is connected ; ; Parameters: GS Hub selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; is_usb_hub_port_connected_name DB 'Is Usb Hub Port Connected', 0 is_usb_hub_port_connected Proc far push ax push si ; mov si,dx dec si shl si,5 mov ax,gs:[si].hub_port_arr.hps_status test al,1 clc jnz iuhDone ; stc iuhDone: pop si pop ax retf32 is_usb_hub_port_connected Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ResetUsbHubPort ; ; description: Reset Hub port ; ; Parameters: GS Hub selector ; DX Port ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; reset_usb_hub_port_name DB 'Reset Usb Hub Port', 0 reset_usb_hub_port Proc far push ax push si ; mov si,dx dec si shl si,5 mov gs:[si].hub_port_arr.hps_ext_reset,1 ; pop si pop ax retf32 reset_usb_hub_port Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: init ; ; description: Init device ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; init Proc far mov bx,SEG data mov ds,bx mov ds:hub_list,0 InitSection ds:hub_section ; mov ax,cs mov ds,ax mov es,ax ; mov esi,OFFSET is_usb_hub_port_connected mov edi,OFFSET is_usb_hub_port_connected_name xor cl,cl mov ax,is_usb_hub_port_connected_nr RegisterOsGate ; mov esi,OFFSET reset_usb_hub_port mov edi,OFFSET reset_usb_hub_port_name xor cl,cl mov ax,reset_usb_hub_port_nr RegisterOsGate ; mov edi,OFFSET usb_attach HookUsbAttach ; mov edi,OFFSET usb_detach HookUsbDetach clc ret init Endp code ENDS END init