;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; 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 ; ; SMPRESP.ASM ; Response handling of Simple Message Protocol (remote IPC) ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; NAME smpresp ;;;;;;;;; INTERNAL PROCEDURES ;;;;;;;;;;; GateSize = 16 INCLUDE protseg.def INCLUDE ..\driver.def INCLUDE ..\user.def INCLUDE ..\os.def INCLUDE ..\user.inc INCLUDE ..\os.inc INCLUDE exec.def INCLUDE system.def INCLUDE system.inc INCLUDE ip.inc INCLUDE ipc.inc INCLUDE smp.inc Reverse MACRO xchg al,ah rol eax,16 xchg al,ah ENDM smp_host_response STRUC shr_link DW ? shr_data smp_response <> smp_host_response ENDS code SEGMENT byte public 'CODE' .386p assume cs:code extrn FindHost:near extrn CalcChecksum:near extrn ReceiverAck:near extrn SenderAck:near extrn FindReceiveMailslotHost:near extrn FindReceiveMailslot:near extrn FindSendMailslot:near extrn HandleReset:near PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; Name: CreateSegment ; ; Purpose: Create a segment ; ; Parameters: FS Host data selector ; CX Size of data ; ; Returns: ES:DI SMP header ; ES:BX SMP data ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; smp_options DB 0 public CreateSegment CreateSegment Proc near push ds push ax push ecx push edx push esi ; mov ax,fs:shd_response_list or ax,ax jz create_segment_no_response ; mov ax,fs mov ds,ax EnterSection ds:shd_section ; push ds mov ax,ds:shd_response_list create_segment_size_loop: or ax,ax jz create_segment_alloc ; mov ds,ax add cx,SIZE smp_response add cl,ds:shr_data.sr_size adc ch,0 mov ax,ds:shr_link jmp create_segment_size_loop create_segment_alloc: mov al,121 mov ah,60 mov bx,fs:shd_id inc bx mov fs:shd_id,bx xchg bl,bh add cx,SIZE smp_header movzx ecx,cx mov edx,fs:shd_ip mov si,cs mov ds,si mov esi,OFFSET smp_options CreateIpHeader pop ds ; mov ax,ds:shd_response_list xor dl,dl push ds push di ; add di,SIZE smp_header create_segment_move_loop: or ax,ax jz create_segment_leave ; mov ds,ax mov cx,SIZE smp_response add cl,ds:shr_data.sr_size adc ch,0 mov si,OFFSET shr_data rep movsb mov ax,ds:shr_link inc dl push es mov cx,ds mov es,cx xor cx,cx mov ds,cx FreeMem pop es jmp create_segment_move_loop create_segment_leave: mov bx,di pop di pop ds mov ds:shd_response_list,0 LeaveSection ds:shd_section mov es:[di].sh_responses,dl jmp create_segment_done create_segment_no_response: mov al,121 mov ah,60 mov bx,fs:shd_id inc bx mov fs:shd_id,bx xchg bl,bh add cx,SIZE smp_header movzx ecx,cx mov edx,fs:shd_ip mov si,cs mov ds,si mov esi,OFFSET smp_options CreateIpHeader mov bx,di add bx,SIZE smp_header mov es:[di].sh_responses,0 create_segment_done: pop esi pop edx pop ecx pop ax pop ds ret CreateSegment Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; Name: FlushResponses ; ; Purpose: Flush all responses ; ; Parameters: FS Host data selector ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; public FlushResponses FlushResponses Proc near push ds push es push ax ; mov ax,fs:shd_response_list or ax,ax jz flush_resp_done ; mov ax,fs mov ds,ax EnterSection ds:shd_section ; mov ax,ds:shd_response_list flush_resp_loop: or ax,ax jz flush_resp_leave ; mov es,ax mov ax,es:shr_link FreeMem jmp flush_resp_loop flush_resp_leave: mov ds:shd_response_list,0 LeaveSection ds:shd_section flush_resp_done: pop ax pop es pop ds ret FlushResponses Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; Name: HandleAck ; ; Purpose: Handle ACK response ; ; Parameters: ES:SI Response data ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; HandleAck Proc near push ds push eax push bx push edx ; cmp es:[si].sr_size,4 jnz handle_ack_done ; mov bx,es:[si].sr_mailslot xchg bl,bh mov edx,es:ip_source call FindSendMailslot jc handle_ack_check_receive ; call SenderAck jmp handle_ack_done handle_ack_check_receive: call FindReceiveMailslot jc handle_ack_done ; mov edx,es:ip_source call FindReceiveMailslotHost jc handle_ack_done ; push gs mov gs,ax call ReceiverAck pop gs handle_ack_done: pop edx pop bx pop eax pop ds ret HandleAck Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; Name: HandleTooLarge ; ; Purpose: Handle too large message ; ; Parameters: ES:SI Response data ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; HandleTooLarge Proc near ret HandleTooLarge Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; Name: HandleBusy ; ; Purpose: Handle busy response ; ; Parameters: ES:SI Response data ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; HandleBusy Proc near int 3 ret HandleBusy Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; Name: ActOnResponse ; ; Purpose: Act on a response ; ; Parameters: ES:SI Response data ; CX Size of data ; ; Returns: ES:SI Next response ; CX Size of data ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ResponseError Proc near ret ResponseError Endp ResponseTab: rt00 DW OFFSET ResponseError rt01 DW OFFSET HandleReset rt02 DW OFFSET HandleAck rt03 DW OFFSET HandleTooLarge rt04 DW OFFSET HandleBusy rt05 DW OFFSET ResponseError rt06 DW OFFSET ResponseError rt07 DW OFFSET ResponseError rt08 DW OFFSET ResponseError rt09 DW OFFSET ResponseError rt0A DW OFFSET ResponseError rt0B DW OFFSET ResponseError rt0C DW OFFSET ResponseError rt0D DW OFFSET ResponseError rt0E DW OFFSET ResponseError rt0F DW OFFSET ResponseError ActOnResponse Proc near push bx ; sub cx,SIZE smp_response jc act_on_response_done sub cl,es:[si].sr_size sbb ch,0 jc act_on_response_done ; movzx bx,es:[si].sr_action cmp bx,10h cmc jc act_on_response_done ; add bx,bx call word ptr cs:[bx].ResponseTab movzx bx,es:[si].sr_size add si,SIZE smp_response add si,bx clc act_on_response_done: pop bx ret ActOnResponse Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; Name: HandleResponses ; ; Purpose: Handle responses ; ; Parameters: ES:DI SMP header ; CX Size of data + header ; ; Returns: ES:SI SMP data ; CX Size of data ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; public HandleResponses HandleResponses Proc near push dx ; mov si,di add si,SIZE smp_header sub cx,SIZE smp_header jc handle_responses_done ; mov dl,es:[di].sh_responses or dl,dl clc jz handle_responses_done handle_responses_loop: call ActOnResponse jc handle_responses_done ; sub dl,1 jnz handle_responses_loop clc handle_responses_done: pop dx ret HandleResponses Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; Name: QueueResponse ; ; Purpose: Queue a response ; ; Parameters: DS Host data selector ; ES Response ; EDX:EAX Response timeout ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; QueueResponse Proc near push bx ; EnterSection ds:shd_section push ax mov ax,ds:shd_response_list mov es:shr_link,ax mov ds:shd_response_list,es xor bx,bx mov es,bx or ax,ax pop ax jz queue_response_set_timeout ; sub eax,ds:shd_response_time sbb edx,ds:shd_response_time+4 jnc queue_response_leave ; add eax,ds:shd_response_time adc edx,ds:shd_response_time+4 queue_response_set_timeout: cli mov ds:shd_response_time,eax mov ds:shd_response_time+4,edx sti queue_response_leave: xor ax,ax mov es,ax LeaveSection ds:shd_section ; pop bx ret QueueResponse Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; Name: QueueReset ; ; Purpose: Queue reset message ; ; Parameters: DS Host ; BX Mailslot ; EDX Connection # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; public QueueReset QueueReset Proc near push es push eax push bx push edx ; mov eax,SIZE smp_host_response AllocateSmallGlobalMem mov eax,edx Reverse mov es:shr_data.sr_connection,eax mov ax,bx xchg al,ah mov es:shr_data.sr_mailslot,ax mov es:shr_data.sr_size,0 mov es:shr_data.sr_action,ACTION_RESET ; GetSystemTime add eax,ACK_DELAY * 1193 adc edx,0 call QueueResponse ; pop edx pop bx pop eax pop es ret QueueReset Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; Name: QueueAck ; ; Purpose: Queue an ACK response ; ; Parameters: DS Host ; BX Mailslot ; ECX Ack position ; EDX Connection ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; public QueueAck QueueAck Proc near push es push eax push bx push edx ; mov eax,SIZE smp_host_response + 4 AllocateSmallGlobalMem mov eax,edx Reverse mov es:shr_data.sr_connection,eax mov ax,bx xchg al,ah mov es:shr_data.sr_mailslot,ax mov es:shr_data.sr_size,4 mov es:shr_data.sr_action,ACTION_ACK mov bx,SIZE smp_host_response mov eax,ecx Reverse mov es:[bx],eax ; GetSystemTime add eax,ACK_DELAY * 1193 adc edx,0 call QueueResponse ; pop edx pop bx pop eax pop es ret QueueAck Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; Name: QueueTooLarge ; ; Purpose: Queue too large message error ; ; Parameters: DS Host ; BX Mailslot ; EDX Connection # ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; public QueueTooLarge QueueTooLarge Proc near push es push eax push bx push edx ; mov eax,SIZE smp_host_response AllocateSmallGlobalMem mov eax,edx Reverse mov es:shr_data.sr_connection,eax mov ax,bx xchg al,ah mov es:shr_data.sr_mailslot,ax mov es:shr_data.sr_size,0 mov es:shr_data.sr_action,ACTION_TOO_LARGE ; GetSystemTime add eax,ACK_DELAY * 1193 adc edx,0 call QueueResponse ; pop edx pop bx pop eax pop es ret QueueTooLarge Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ResponseSupervise ; ; DESCRIPTION: Supervise response operations ; ; PARAMETERS: FS Host ; GS IPC data sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; public ResponseSupervise ResponseSupervise Proc near mov ax,fs:shd_response_list or ax,ax jz response_supervise_done ; GetSystemTime sub eax,fs:shd_response_time sbb edx,fs:shd_response_time+4 jc response_supervise_done ; mov ax,gs:super_response_list mov fs:shd_perform_list,ax mov gs:super_response_list,fs response_supervise_done: ret ResponseSupervise Endp PAGE ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: SendResponses ; ; DESCRIPTION: Send responses through SMP ; ; PARAMETERS: FS Host ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; SendResponses Proc near xor ecx,ecx call CreateSegment ; xor eax,eax mov es:[di].sh_connection,eax mov es:[di].sh_offset_size,eax mov es:[di].sh_mailslot,ax mov es:[di].sh_flags,al mov es:[di].sh_checksum,ax add cx,bx sub cx,di mov ax,cx xchg al,ah mov es:[di].sh_size,ax ; add ax,7900h adc ax,0 adc ax,0 sub di,8 add cx,8 call CalcChecksum not ax add di,8 mov es:[di].sh_checksum,ax SendIp ret SendResponses Endp ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ; ; ; NAME: ResponsePerform ; ; DESCRIPTION: Perform response operations ; ; PARAMETERS: GS IPC data sel ; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; public ResponsePerform ResponsePerform Proc near mov ax,gs:super_response_list response_perform_loop: or ax,ax jz response_perform_done ; mov fs,ax call SendResponses mov ax,fs:shd_perform_list jmp response_perform_loop response_perform_done: xor ax,ax mov fs,ax mov gs:super_response_list,ax ret ResponsePerform Endp code ENDS END