Small bugfixes and improvements for FTP client
git-svn-id: svn://kolibrios.org@3789 a494cfbc-eb01-0410-851d-a64ba20cac60
This commit is contained in:
parent
3e750a39ab
commit
75cac16296
@ -14,8 +14,6 @@
|
||||
|
||||
format binary as ""
|
||||
|
||||
__DEBUG__ = 0
|
||||
__DEBUG_LEVEL__ = 1
|
||||
BUFFERSIZE = 1024
|
||||
|
||||
STATUS_CONNECTING = 0
|
||||
@ -38,7 +36,6 @@ include '../../macros.inc'
|
||||
purge mov,add,sub
|
||||
include '../../proc32.inc'
|
||||
include '../../dll.inc'
|
||||
include '../../debug-fdo.inc'
|
||||
include '../../network.inc'
|
||||
|
||||
include 'usercommands.inc'
|
||||
@ -47,39 +44,32 @@ include 'servercommands.inc'
|
||||
; entry point
|
||||
start:
|
||||
|
||||
DEBUGF 1, "hello"
|
||||
mcall 40, 0
|
||||
; load libraries
|
||||
stdcall dll.Load, @IMPORT
|
||||
test eax, eax
|
||||
jnz exit
|
||||
; initialize console
|
||||
push 1
|
||||
call [con_start]
|
||||
push title
|
||||
push 25
|
||||
push 80
|
||||
push 25
|
||||
push 80
|
||||
call [con_init]
|
||||
invoke con_start, 1
|
||||
invoke con_init, 80, 25, 80, 25, title
|
||||
|
||||
; Check for parameters
|
||||
cmp byte [s], 0
|
||||
jne resolve
|
||||
|
||||
main:
|
||||
call [con_cls]
|
||||
invoke con_cls
|
||||
; Welcome user
|
||||
push str1
|
||||
call [con_write_asciiz]
|
||||
invoke con_write_asciiz, str1
|
||||
|
||||
; write prompt
|
||||
push str2
|
||||
call [con_write_asciiz]
|
||||
invoke con_set_flags, 0x0a
|
||||
invoke con_write_asciiz, str2
|
||||
; read string
|
||||
mov esi, s
|
||||
push 256
|
||||
push esi
|
||||
call [con_gets]
|
||||
invoke con_gets, esi, 256
|
||||
invoke con_write_asciiz, str4 ; newline
|
||||
invoke con_set_flags, 0x07
|
||||
; check for exit
|
||||
test eax, eax
|
||||
jz done
|
||||
@ -87,7 +77,6 @@ main:
|
||||
jz done
|
||||
|
||||
resolve:
|
||||
|
||||
; delete terminating '\n'
|
||||
mov esi, s
|
||||
@@:
|
||||
@ -96,120 +85,90 @@ resolve:
|
||||
ja @r
|
||||
mov byte [esi-1], 0
|
||||
|
||||
; call [con_cls]
|
||||
push str3
|
||||
call [con_write_asciiz]
|
||||
push s
|
||||
call [con_write_asciiz]
|
||||
invoke con_write_asciiz, str3
|
||||
invoke con_write_asciiz, s
|
||||
|
||||
; resolve name
|
||||
push esp ; reserve stack place
|
||||
push esp ; fourth parameter
|
||||
push 0 ; third parameter
|
||||
push 0 ; second parameter
|
||||
push s ; first parameter
|
||||
call [getaddrinfo]
|
||||
invoke getaddrinfo, s, 0, 0, esp
|
||||
pop esi
|
||||
; test for error
|
||||
test eax, eax
|
||||
jnz fail
|
||||
|
||||
; write results
|
||||
push str8
|
||||
call [con_write_asciiz]
|
||||
invoke con_write_asciiz, str8
|
||||
; mov edi, esi
|
||||
|
||||
; convert IP address to decimal notation
|
||||
mov eax, [esi+addrinfo.ai_addr]
|
||||
mov eax, [eax+sockaddr_in.sin_addr]
|
||||
mov [sockaddr1.ip], eax
|
||||
push eax
|
||||
call [inet_ntoa]
|
||||
|
||||
invoke inet_ntoa, eax
|
||||
; write result
|
||||
push eax
|
||||
call [con_write_asciiz]
|
||||
invoke con_write_asciiz, eax
|
||||
; free allocated memory
|
||||
push esi
|
||||
call [freeaddrinfo]
|
||||
|
||||
push str9
|
||||
call [con_write_asciiz]
|
||||
invoke freeaddrinfo, esi
|
||||
|
||||
invoke con_write_asciiz, str9
|
||||
mcall socket, AF_INET4, SOCK_STREAM, 0
|
||||
cmp eax, -1
|
||||
je fail2
|
||||
je socket_error
|
||||
mov [socketnum], eax
|
||||
|
||||
push str11
|
||||
call [con_write_asciiz]
|
||||
|
||||
invoke con_write_asciiz, str11
|
||||
mcall connect, [socketnum], sockaddr1, 18
|
||||
|
||||
mcall 40, EVM_STACK
|
||||
|
||||
mov [status], STATUS_CONNECTING
|
||||
mov [offset], buffer_ptr
|
||||
|
||||
push str12
|
||||
call [con_write_asciiz]
|
||||
invoke con_write_asciiz, str12
|
||||
|
||||
wait_for_serverdata:
|
||||
mcall 10
|
||||
wait_for_servercommand:
|
||||
|
||||
call [con_get_flags]
|
||||
test eax, 0x200 ; con window closed?
|
||||
jnz exit
|
||||
; invoke con_write_asciiz, str_dbg
|
||||
|
||||
; receive socket data
|
||||
mcall recv, [socketnum], [offset], BUFFERSIZE, MSG_DONTWAIT
|
||||
mcall recv, [socketnum], buffer_ptr, BUFFERSIZE, 0
|
||||
inc eax
|
||||
jz wait_for_serverdata
|
||||
jz socket_error
|
||||
dec eax
|
||||
jz wait_for_serverdata
|
||||
jz wait_for_servercommand
|
||||
|
||||
; invoke con_write_asciiz, str_dbg2
|
||||
|
||||
; extract commands, copy them to "s" buffer
|
||||
add eax, buffer_ptr ; eax = end pointer
|
||||
lea ecx, [eax + buffer_ptr] ; ecx = end pointer
|
||||
mov esi, buffer_ptr ; esi = current pointer
|
||||
.nextcommand:
|
||||
mov edi, s
|
||||
.byteloop:
|
||||
cmp esi, eax
|
||||
jae wait_for_serverdata
|
||||
cmp esi, ecx
|
||||
jae wait_for_servercommand
|
||||
lodsb
|
||||
cmp al, 10 ; excellent, we might have a command
|
||||
je .got_command
|
||||
cmp al, 13 ; just ignore this crap
|
||||
cmp al, 13 ; just ignore this byte
|
||||
je .byteloop
|
||||
stosb
|
||||
jmp .byteloop
|
||||
|
||||
; we have a newline check if its a command
|
||||
.got_command:
|
||||
.got_command: ; we have a newline check if its a command
|
||||
xor al, al
|
||||
stosb
|
||||
; push esi eax
|
||||
|
||||
; print it to the screen
|
||||
pushd s
|
||||
call [con_write_asciiz]
|
||||
pushd str4 ; newline
|
||||
call [con_write_asciiz]
|
||||
|
||||
; cmp byte[s+2], " "
|
||||
; jne .not_command
|
||||
|
||||
lea ecx, [edi - s]
|
||||
call server_parser
|
||||
|
||||
; .not_command:
|
||||
; pop eax esi
|
||||
; jmp .nextcommand
|
||||
|
||||
sub edi, s ; length
|
||||
push edi
|
||||
|
||||
invoke con_set_flags, 0x03 ; change color
|
||||
invoke con_write_asciiz, s ; print servercommand
|
||||
invoke con_write_asciiz, str4 ; newline
|
||||
invoke con_set_flags, 0x07
|
||||
|
||||
pop ecx
|
||||
jmp server_parser ; parse command
|
||||
|
||||
wait_for_usercommand:
|
||||
|
||||
invoke con_set_flags, 0x0a
|
||||
|
||||
cmp [status], STATUS_CONNECTED
|
||||
je .connected
|
||||
|
||||
@ -217,17 +176,11 @@ wait_for_usercommand:
|
||||
je .needpass
|
||||
|
||||
; write prompt
|
||||
push str2
|
||||
call [con_write_asciiz]
|
||||
invoke con_write_asciiz, str2
|
||||
; read string
|
||||
mov esi, s
|
||||
push 256
|
||||
push esi
|
||||
call [con_gets]
|
||||
|
||||
call [con_get_flags]
|
||||
test eax, 0x200 ; con window closed?
|
||||
jnz exit
|
||||
invoke con_gets, esi, 256
|
||||
invoke con_set_flags, 0x07
|
||||
|
||||
cmp dword[s], "list"
|
||||
je cmd_list
|
||||
@ -235,54 +188,39 @@ wait_for_usercommand:
|
||||
cmp dword[s], "help"
|
||||
je cmd_help
|
||||
|
||||
push str_unkown
|
||||
call [con_write_asciiz]
|
||||
|
||||
invoke con_write_asciiz, str_unknown
|
||||
jmp wait_for_usercommand
|
||||
|
||||
|
||||
.connected:
|
||||
|
||||
push str_user
|
||||
call [con_write_asciiz]
|
||||
|
||||
invoke con_write_asciiz, str_user
|
||||
mov dword[s], "USER"
|
||||
mov byte[s+4], " "
|
||||
|
||||
; mov [status], STATUS_NEEDPASSWORD
|
||||
inc [status]
|
||||
|
||||
jmp .send
|
||||
|
||||
|
||||
.needpass:
|
||||
push str_pass
|
||||
call [con_write_asciiz]
|
||||
|
||||
invoke con_write_asciiz, str_pass
|
||||
mov dword[s], "PASS"
|
||||
mov byte[s+4], " "
|
||||
|
||||
; mov [status], STATUS_LOGGED_IN
|
||||
inc [status]
|
||||
|
||||
.send:
|
||||
; read string
|
||||
mov esi, s+5
|
||||
push 256
|
||||
push esi
|
||||
call [con_gets]
|
||||
invoke con_gets, esi, 256
|
||||
|
||||
mov edi, s+5
|
||||
mov ecx, 256
|
||||
xor al, al
|
||||
repne scasb
|
||||
lea esi, [edi-s-1]
|
||||
mcall send, [socketnum], s
|
||||
|
||||
jmp wait_for_usercommand
|
||||
|
||||
|
||||
mcall send, [socketnum], s, , 0
|
||||
|
||||
invoke con_write_asciiz, str4 ; newline
|
||||
invoke con_set_flags, 0x07
|
||||
jmp wait_for_servercommand
|
||||
|
||||
|
||||
|
||||
@ -292,37 +230,30 @@ open_dataconnection:
|
||||
|
||||
mov dword[s], "PASV"
|
||||
mov byte[s+4], 10
|
||||
mcall send, [socketnum], s, 5
|
||||
|
||||
mcall send, [socketnum], s, 5, 0
|
||||
ret
|
||||
|
||||
.fail:
|
||||
push str6
|
||||
call [con_write_asciiz]
|
||||
|
||||
invoke con_write_asciiz, str6
|
||||
ret
|
||||
|
||||
|
||||
fail2:
|
||||
push str6
|
||||
call [con_write_asciiz]
|
||||
|
||||
socket_error:
|
||||
invoke con_write_asciiz, str6
|
||||
jmp fail.wait
|
||||
|
||||
fail:
|
||||
push str5
|
||||
call [con_write_asciiz]
|
||||
invoke con_write_asciiz, str5
|
||||
.wait:
|
||||
push str10
|
||||
call [con_write_asciiz]
|
||||
call [con_getch2]
|
||||
invoke con_write_asciiz, str10
|
||||
invoke con_getch2
|
||||
jmp main
|
||||
|
||||
done:
|
||||
push 1
|
||||
call [con_exit]
|
||||
exit:
|
||||
invoke con_exit, 1
|
||||
|
||||
exit:
|
||||
mcall close, [socketnum]
|
||||
mcall -1
|
||||
|
||||
@ -330,7 +261,7 @@ exit:
|
||||
|
||||
; data
|
||||
title db 'FTP client',0
|
||||
str1 db 'FTP client for KolibriOS v0.01',10,10,'Please enter ftp server address.',10,0
|
||||
str1 db 'FTP client for KolibriOS v0.02',10,10,'Please enter ftp server address.',10,0
|
||||
str2 db '> ',0
|
||||
str3 db 'Resolving ',0
|
||||
str4 db 10,0
|
||||
@ -339,16 +270,19 @@ str6 db 10,'Socket error.',10,0
|
||||
str8 db ' (',0
|
||||
str9 db ')',10,0
|
||||
str10 db 'Push any key to continue.',0
|
||||
str11 db 'Connecting',10,0
|
||||
str12 db 'Connected!',10,0
|
||||
str11 db 'Connecting...',10,0
|
||||
str12 db 'Waiting for welcome message.',10,0
|
||||
str_user db "username: ",0
|
||||
str_pass db "password: ",0
|
||||
str_unkown db "unkown command",10,0
|
||||
str_unknown db "unknown command",10,0
|
||||
str_help db "available commands:",10,10
|
||||
db "help list",10,0
|
||||
|
||||
str_open db "opening data socket",10,0
|
||||
|
||||
str_dbg db 'debug',10,0
|
||||
str_dbg2 db 'debug2',10,0
|
||||
|
||||
sockaddr1:
|
||||
dw AF_INET4
|
||||
.port dw 0x1500 ; 21
|
||||
@ -361,9 +295,6 @@ sockaddr2:
|
||||
.ip dd 0
|
||||
rb 10
|
||||
|
||||
include_debug_strings ; ALWAYS present in data section
|
||||
|
||||
|
||||
|
||||
; import
|
||||
align 4
|
||||
@ -386,7 +317,8 @@ import console, \
|
||||
con_getch2, 'con_getch2',\
|
||||
con_set_cursor_pos, 'con_set_cursor_pos',\
|
||||
con_write_string, 'con_write_string',\
|
||||
con_get_flags, 'con_get_flags'
|
||||
con_get_flags, 'con_get_flags', \
|
||||
con_set_flags, 'con_set_flags'
|
||||
|
||||
|
||||
i_end:
|
||||
@ -396,7 +328,6 @@ socketnum dd ?
|
||||
datasocket dd ?
|
||||
buffer_ptr rb 2*BUFFERSIZE
|
||||
status db ?
|
||||
offset dd ?
|
||||
|
||||
s rb 1024
|
||||
|
||||
|
@ -3,7 +3,7 @@ server_parser:
|
||||
; Commands are always 3 numbers and followed by a space
|
||||
; If a server decides it needs multiline output,
|
||||
; first lines will have a dash instead of space after numbers,
|
||||
; thus they are simply ignored.
|
||||
; thus they are simply ignored in this simple command parser.
|
||||
|
||||
cmp dword[s], "150 "
|
||||
je data_ok
|
||||
@ -11,6 +11,9 @@ server_parser:
|
||||
cmp dword[s], "220 "
|
||||
je welcome
|
||||
|
||||
cmp dword[s], "226 "
|
||||
; je list_ok
|
||||
|
||||
cmp dword[s], "227 "
|
||||
je pasv_ok
|
||||
|
||||
@ -20,25 +23,31 @@ server_parser:
|
||||
cmp dword[s], "331 "
|
||||
je pass
|
||||
|
||||
ret
|
||||
cmp dword[s], "421 "
|
||||
; je timeout
|
||||
|
||||
cmp dword[s], "530" ; password incorrect
|
||||
je welcome
|
||||
|
||||
jmp wait_for_usercommand
|
||||
|
||||
|
||||
welcome:
|
||||
|
||||
mov [status], STATUS_CONNECTED
|
||||
ret
|
||||
jmp wait_for_usercommand
|
||||
|
||||
|
||||
pass:
|
||||
|
||||
mov [status], STATUS_NEEDPASSWORD
|
||||
ret
|
||||
jmp wait_for_usercommand
|
||||
|
||||
|
||||
login_ok:
|
||||
|
||||
mov [status], STATUS_LOGGED_IN
|
||||
ret
|
||||
jmp wait_for_usercommand
|
||||
|
||||
|
||||
pasv_ok:
|
||||
@ -64,18 +73,16 @@ pasv_ok:
|
||||
call ascii_dec
|
||||
mov byte[sockaddr2.ip+3], bl
|
||||
|
||||
call ascii_dec
|
||||
mov byte[sockaddr2.port+1], bl
|
||||
call ascii_dec
|
||||
mov byte[sockaddr2.port+0], bl
|
||||
call ascii_dec
|
||||
mov byte[sockaddr2.port+1], bl
|
||||
|
||||
push str_open
|
||||
call [con_write_asciiz]
|
||||
|
||||
invoke con_write_asciiz, str_open
|
||||
mcall connect, [datasocket], sockaddr2, 18
|
||||
|
||||
.fail:
|
||||
ret
|
||||
jmp wait_for_servercommand
|
||||
|
||||
|
||||
data_ok:
|
||||
@ -85,27 +92,27 @@ data_ok:
|
||||
jz .fail
|
||||
dec eax
|
||||
jz .fail
|
||||
|
||||
mov byte[buffer_ptr + eax], 0
|
||||
pushd buffer_ptr
|
||||
call [con_write_asciiz]
|
||||
|
||||
invoke con_write_asciiz, buffer_ptr
|
||||
|
||||
.fail:
|
||||
ret
|
||||
mcall close, [datasocket]
|
||||
jmp wait_for_servercommand
|
||||
|
||||
|
||||
ascii_dec:
|
||||
|
||||
xor ebx, ebx
|
||||
mov cl, 3
|
||||
mov cl, 4 ; max length is 3 digits + 1 separator
|
||||
.loop:
|
||||
lodsb
|
||||
sub al, '0'
|
||||
jb .done
|
||||
cmp al, 9
|
||||
ja .done
|
||||
lea ebx, [ebx*4+ebx]
|
||||
shl ebx, 1
|
||||
lea ebx, [ebx*4+ebx] ; ebx *5
|
||||
shl ebx, 1 ; ebx *2
|
||||
add bl, al
|
||||
dec cl
|
||||
jnz .loop
|
||||
|
@ -3,10 +3,10 @@ cmd_list:
|
||||
call open_dataconnection
|
||||
|
||||
mov dword[s], "LIST"
|
||||
mov word[s+4], 0x0d0a
|
||||
mcall send, [socketnum], s, 6
|
||||
mov byte[s+4], 0x0a
|
||||
mcall send, [socketnum], s, 5
|
||||
|
||||
jmp wait_for_serverdata
|
||||
jmp wait_for_servercommand
|
||||
|
||||
|
||||
cmd_help:
|
||||
|
Loading…
Reference in New Issue
Block a user