Andrew Dent
2d3e0ac7f4
- Corrections for en_US language. - Some whitespace sanitation. git-svn-id: svn://kolibrios.org@10061 a494cfbc-eb01-0410-851d-a64ba20cac60
1289 lines
33 KiB
PHP
1289 lines
33 KiB
PHP
|
|
|
|
struct thread_data
|
|
rb 1024
|
|
stack rb 0
|
|
|
|
home_dir rb 1024 ; home directory in which the user is locked, asciiz
|
|
work_dir rb 1024 ; working directory, must at all times begin and end with a '/', asciiz
|
|
fpath rb 1024*3 ; file path, combination of home_dir, work_dir and filename
|
|
; Will also be used to temporarily store username
|
|
|
|
type db ? ; ASCII/EBDIC/IMAGE/..
|
|
mode db ? ; active/passive
|
|
socketnum dd ? ; Commands socket
|
|
state dd ? ; disconnected/logging in/logged in/..
|
|
passivesocknum dd ? ; when in passive mode, this is the listening socket
|
|
datasocketnum dd ? ; socket used for data transfers
|
|
permissions dd ? ; read/write/execute/....
|
|
buffer_ptr dd ?
|
|
pid dd ? ; Process id of the current thread
|
|
|
|
datasock sockaddr_in
|
|
|
|
buffer rb BUFFERSIZE
|
|
ends
|
|
|
|
;------------------------------------------------
|
|
; parse_cmd
|
|
;
|
|
; Internal function which uses the 'commands'
|
|
; table to call an appropriate cmd_xx function.
|
|
;
|
|
; input: esi = ptr to ascii commands
|
|
; ecx = number of bytes input
|
|
; ebp = pointer to thread_data structure
|
|
;
|
|
; output: none
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
parse_cmd: ; esi must point to command
|
|
|
|
cmp byte [esi], 0x20 ; skip all leading characters
|
|
ja .ok
|
|
inc esi
|
|
dec ecx
|
|
cmp ecx, 3
|
|
jb .error
|
|
jmp parse_cmd
|
|
.ok:
|
|
cmp byte [esi+3], 0x20
|
|
ja @f
|
|
mov byte [esi+3], 0
|
|
@@:
|
|
|
|
mov eax, [esi]
|
|
and eax, not 0x20202020 ; convert to upper case
|
|
mov edi, commands ; list of commands to scan
|
|
.scanloop:
|
|
cmp eax, [edi]
|
|
je .got_it
|
|
|
|
add edi, 5*4
|
|
cmp byte [edi], 0
|
|
jne .scanloop
|
|
|
|
.error:
|
|
cmp [ebp + thread_data.state], STATE_ACTIVE
|
|
jb login_first
|
|
sendFTP "500 Unsupported command"
|
|
ret
|
|
|
|
.got_it:
|
|
mov eax, [ebp + thread_data.state]
|
|
jmp dword [edi + 4 + eax]
|
|
|
|
|
|
align 4
|
|
iglobal
|
|
commands: ; all commands must be in uppercase
|
|
|
|
dd 'ABOR', login_first, login_first, login_first, cmdABOR
|
|
; dd 'ACCT', login_first, login_first, login_first, cmd_ACCT
|
|
; dd 'APPE', login_first, login_first, login_first, cmd_APPE
|
|
dd 'CDUP', login_first, login_first, login_first, cmdCDUP
|
|
dd 'CWD', login_first, login_first, login_first, cmdCWD
|
|
dd 'DELE', login_first, login_first, login_first, cmdDELE
|
|
; dd 'HELP', login_first, login_first, login_first, cmd_HELP
|
|
dd 'LIST', login_first, login_first, login_first, cmdLIST
|
|
; dd 'MDTM', login_first, login_first, login_first, cmd_MDTM
|
|
; dd 'MKD', login_first, login_first, login_first, cmd_MKD
|
|
; dd 'MODE', login_first, login_first, login_first, cmd_MODE
|
|
; dd 'NLST', login_first, login_first, login_first, cmdNLST
|
|
dd 'NOOP', login_first, login_first, login_first, cmdNOOP
|
|
dd 'PASS', cmdPASS.0, cmdPASS , cmdPASS.2, cmdPASS.3
|
|
dd 'PASV', login_first, login_first, login_first, cmdPASV
|
|
dd 'PORT', login_first, login_first, login_first, cmdPORT
|
|
dd 'PWD', login_first, login_first, login_first, cmdPWD
|
|
dd 'QUIT', cmdQUIT, cmdQUIT, cmdQUIT, cmdQUIT
|
|
; dd 'REIN', login_first, login_first, login_first, cmd_REIN
|
|
; dd 'REST', login_first, login_first, login_first, cmd_REST
|
|
dd 'RETR', login_first, login_first, login_first, cmdRETR
|
|
; dd 'RMD', login_first, login_first, login_first, cmd_RMD
|
|
; dd 'RNFR', login_first, login_first, login_first, cmd_RNFR
|
|
; dd 'RNTO', login_first, login_first, login_first, cmd_RNTO
|
|
; dd 'SITE', login_first, login_first, login_first, cmd_SITE
|
|
; dd 'SIZE', login_first, login_first, login_first, cmd_SIZE
|
|
; dd 'STAT', login_first, login_first, login_first, cmd_STAT
|
|
dd 'STOR', login_first, login_first, login_first, cmdSTOR
|
|
; dd 'STOU', login_first, login_first, login_first, cmd_STOU
|
|
; dd 'STRU', login_first, login_first, login_first, cmd_STRU
|
|
dd 'SYST', login_first, login_first, login_first, cmdSYST
|
|
dd 'TYPE', login_first, login_first, login_first, cmdTYPE
|
|
dd 'USER', cmdUSER, cmdUSER, cmdUSER, cmdUSER.2
|
|
db 0 ; end marker
|
|
endg
|
|
|
|
align 4
|
|
login_first:
|
|
sendFTP "530 Please login with USER and PASS"
|
|
ret
|
|
|
|
align 4
|
|
permission_denied:
|
|
sendFTP "550 Permission denied"
|
|
ret
|
|
|
|
align 4
|
|
socketerror:
|
|
invoke con_set_flags, 0x0c
|
|
invoke con_write_asciiz, str_sockerr
|
|
invoke con_set_flags, 0x07
|
|
|
|
sendFTP "425 Can't open data connection"
|
|
ret
|
|
|
|
align 4
|
|
abort_transfer:
|
|
and [ebp + thread_data.permissions], not ABORT
|
|
mov [ebp + thread_data.mode], MODE_NOTREADY
|
|
invoke file.close, ebx
|
|
mcall close, [ebp + thread_data.datasocketnum]
|
|
|
|
sendFTP "530 Transfer aborted"
|
|
ret
|
|
|
|
align 4
|
|
ip_to_dword: ; esi = ptr to str, cl = separator ('.', ',')
|
|
|
|
call ascii_to_byte
|
|
mov bl, al
|
|
cmp byte [esi], cl
|
|
jne .err
|
|
inc esi
|
|
|
|
call ascii_to_byte
|
|
mov bh, al
|
|
cmp byte [esi], cl
|
|
jne .err
|
|
inc esi
|
|
|
|
shl ebx, 16
|
|
|
|
call ascii_to_byte
|
|
mov bl, al
|
|
cmp byte [esi], cl
|
|
jne .err
|
|
inc esi
|
|
|
|
call ascii_to_byte
|
|
mov bh, al
|
|
|
|
ror ebx, 16
|
|
ret
|
|
|
|
.err:
|
|
xor ebx, ebx
|
|
ret
|
|
|
|
align 4 ; esi = ptr to str, output in eax
|
|
ascii_to_byte:
|
|
|
|
xor eax, eax
|
|
push ebx
|
|
|
|
.loop:
|
|
movzx ebx, byte[esi]
|
|
sub bl, '0'
|
|
jb .done
|
|
cmp bl, 9
|
|
ja .done
|
|
lea eax, [eax*4 + eax] ;
|
|
shl eax, 1 ; eax = eax * 10
|
|
add eax, ebx
|
|
inc esi
|
|
|
|
jmp .loop
|
|
|
|
.done:
|
|
pop ebx
|
|
ret
|
|
|
|
align 4
|
|
dword_to_ascii: ; edi = ptr where to write, eax is number
|
|
|
|
push edx ebx ecx
|
|
mov ebx, 10
|
|
xor ecx, ecx
|
|
|
|
@@:
|
|
xor edx, edx
|
|
div ebx
|
|
add edx, '0'
|
|
pushw dx
|
|
inc ecx
|
|
test eax, eax
|
|
jnz @r
|
|
|
|
@@:
|
|
popw ax
|
|
stosb
|
|
dec ecx
|
|
jnz @r
|
|
|
|
pop ecx ebx edx
|
|
ret
|
|
|
|
align 4
|
|
create_path: ; combine home_dir and work_dir strings into fpath
|
|
|
|
lea edi, [ebp + thread_data.fpath]
|
|
lea esi, [ebp + thread_data.home_dir]
|
|
mov ecx, 1024
|
|
.loop1:
|
|
lodsb
|
|
cmp al, 0x20
|
|
jb .next
|
|
stosb
|
|
loop .loop1
|
|
.next:
|
|
|
|
cmp byte[edi-1], '/'
|
|
jne @f
|
|
dec edi
|
|
@@:
|
|
|
|
lea esi, [ebp + thread_data.work_dir]
|
|
mov ecx, 1024
|
|
.loop2:
|
|
lodsb
|
|
cmp al, 0x20
|
|
jb .done
|
|
stosb
|
|
loop .loop2
|
|
|
|
.done:
|
|
xor al, al
|
|
stosb
|
|
ret
|
|
|
|
|
|
align 4
|
|
nextpasvport:
|
|
|
|
inc [pasv_port]
|
|
|
|
mov ax, [pasv_port]
|
|
cmp ax, [pasv_start]
|
|
jb .restart
|
|
cmp ax, [pasv_end]
|
|
ja .restart
|
|
|
|
ret
|
|
|
|
.restart:
|
|
pushw [pasv_start]
|
|
popw [pasv_port]
|
|
|
|
ret
|
|
|
|
|
|
align 4
|
|
open_datasock:
|
|
|
|
cmp [ebp + thread_data.mode], MODE_PASSIVE_OK
|
|
je .already_open
|
|
|
|
; If we are in active mode, it's time to open the data socket..
|
|
cmp [ebp + thread_data.mode], MODE_ACTIVE
|
|
jne .not_active
|
|
mov ecx, [ebp + thread_data.datasocketnum]
|
|
lea edx, [ebp + thread_data.datasock]
|
|
mov esi, sizeof.thread_data.datasock
|
|
mcall connect
|
|
cmp eax, -1
|
|
je .socketerror
|
|
invoke con_write_asciiz, str_datasock2
|
|
ret
|
|
|
|
.already_open:
|
|
invoke con_write_asciiz, str_alopen
|
|
ret
|
|
|
|
.socketerror:
|
|
add esp, 4
|
|
jmp socketerror
|
|
|
|
; If we are still in passive_wait, it's time we accept an incoming call...
|
|
.not_active:
|
|
cmp [ebp + thread_data.mode], MODE_PASSIVE_WAIT
|
|
jne .socketerror
|
|
|
|
.try_now:
|
|
mov ecx, [ebp + thread_data.passivesocknum]
|
|
lea edx, [ebp + thread_data.datasock]
|
|
mov esi, sizeof.thread_data.datasock
|
|
mcall accept
|
|
cmp eax, -1
|
|
jne .pasv_ok
|
|
mov [ebp + thread_data.mode], MODE_PASSIVE_FAILED ; assume that we will fail
|
|
mcall 23, 200
|
|
mcall accept
|
|
cmp eax, -1
|
|
je .socketerror
|
|
.pasv_ok:
|
|
mov [ebp + thread_data.datasocketnum], eax
|
|
mov [ebp + thread_data.mode], MODE_PASSIVE_OK
|
|
mcall close ; [ebp + thread_data.passivesocknum]
|
|
mov [ebp + thread_data.passivesocknum], -1
|
|
invoke con_write_asciiz, str_datasock
|
|
|
|
ret
|
|
|
|
|
|
;------------------------------------------------
|
|
; "ABOR"
|
|
;
|
|
; This command aborts the current filetransfer.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdABOR:
|
|
|
|
or [ebp + thread_data.permissions], ABORT
|
|
sendFTP "250 Command succesul"
|
|
ret
|
|
|
|
;------------------------------------------------
|
|
; "CDUP"
|
|
;
|
|
; Change the directory to move up one level.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdCDUP:
|
|
|
|
test [ebp + thread_data.permissions], PERMISSION_CD
|
|
jz permission_denied
|
|
|
|
cmp byte [ebp + thread_data.work_dir+1], 0 ; are we in "/" ?
|
|
je .done ; if so, we can't go up.
|
|
|
|
; find the end of asciiz string work_dir
|
|
mov ecx, 1024
|
|
xor al, al
|
|
lea edi, [ebp + thread_data.work_dir]
|
|
repne scasb
|
|
; return 2 characters (right before last /)
|
|
sub edi, 3
|
|
; and now search backwards, for a '/'
|
|
mov al,'/'
|
|
neg ecx
|
|
add ecx, 1024
|
|
std
|
|
repne scasb
|
|
cld
|
|
; terminate the string here
|
|
mov byte[edi+2], 0
|
|
|
|
.done:
|
|
; Print the new working dir on the console
|
|
lea eax, [ebp + thread_data.work_dir]
|
|
invoke con_write_asciiz, eax
|
|
invoke con_write_asciiz, str_newline
|
|
|
|
sendFTP "250 Command succesul"
|
|
ret
|
|
|
|
;------------------------------------------------
|
|
; "CWD"
|
|
;
|
|
; Change Working Directory.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdCWD:
|
|
|
|
test [ebp + thread_data.permissions], PERMISSION_CD
|
|
jz permission_denied
|
|
|
|
; do we have enough parameters?
|
|
sub ecx, 4
|
|
jbe .err
|
|
|
|
; get ready to copy the path
|
|
add esi, 4
|
|
mov ecx, 1024
|
|
lea edi, [ebp + thread_data.work_dir]
|
|
|
|
; if received dir starts with '/', we will simply copy it
|
|
; If not, we will append the current path with the received path.
|
|
cmp byte [esi], '/'
|
|
je .copyloop
|
|
|
|
; Find the end of work_dir string.
|
|
xor al, al
|
|
.find_zero:
|
|
repne scasb
|
|
dec edi
|
|
|
|
; and now append work_dir with received string
|
|
mov ecx, 1024
|
|
|
|
; scan for end byte, or '.'
|
|
.copyloop:
|
|
lodsb
|
|
cmp al, 0x20
|
|
jb .done
|
|
;;; cmp al, '.' ; '..' means we must go up one dir TODO
|
|
;;; je .up
|
|
stosb
|
|
loop .copyloop
|
|
|
|
; now, now make sure it ends with '/', 0
|
|
.done:
|
|
cmp byte [edi-1], '/'
|
|
je @f
|
|
mov byte [edi], '/'
|
|
inc edi
|
|
@@:
|
|
mov byte [edi], 0
|
|
|
|
; TODO: Check directory on disk
|
|
|
|
|
|
; Print the new working dir on the console
|
|
lea eax, [ebp + thread_data.work_dir]
|
|
invoke con_write_asciiz, eax
|
|
invoke con_write_asciiz, str_newline
|
|
|
|
sendFTP "250 Command succesful"
|
|
ret
|
|
|
|
.err:
|
|
sendFTP "550 Directory does not exist"
|
|
ret
|
|
|
|
;------------------------------------------------
|
|
; "DELE"
|
|
;
|
|
; Delete a file from the server.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdDELE:
|
|
|
|
test [ebp + thread_data.permissions], PERMISSION_DELETE
|
|
jz permission_denied
|
|
|
|
; Create path
|
|
cmp ecx, 1024 + 5
|
|
jae .err
|
|
|
|
sub ecx, 5
|
|
jb .err
|
|
|
|
call create_path
|
|
dec edi
|
|
lea esi, [ebp + thread_data.buffer + 5]
|
|
mov ecx, 1024
|
|
cmp byte [esi], '/'
|
|
jne .loop
|
|
inc esi
|
|
.loop:
|
|
lodsb
|
|
cmp al, 0x20
|
|
jl .done
|
|
stosb
|
|
loop .loop
|
|
.done:
|
|
xor al, al
|
|
stosb
|
|
|
|
lea ebx, [ebp + thread_data.fpath]
|
|
invoke con_write_asciiz, ebx
|
|
invoke con_write_asciiz, str_newline
|
|
; called fs function
|
|
push ebx
|
|
dec esp
|
|
mov byte[esp], 0
|
|
push dword 0
|
|
push dword 0
|
|
push dword 0
|
|
push dword 0
|
|
push dword 8
|
|
mov ebx, esp
|
|
mcall 70
|
|
add esp, 6*4 + 1
|
|
|
|
test eax, eax
|
|
jnz .err
|
|
|
|
sendFTP "250 Command succesful"
|
|
ret
|
|
.err:
|
|
sendFTP "550 No such file"
|
|
ret
|
|
|
|
;------------------------------------------------
|
|
; "LIST"
|
|
;
|
|
; List the files in the current working directory.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdLIST:
|
|
|
|
test [ebp + thread_data.permissions], PERMISSION_EXEC
|
|
jz permission_denied
|
|
|
|
call open_datasock
|
|
|
|
; Create fpath from home_dir and work_dir
|
|
call create_path
|
|
|
|
lea ebx, [ebp + thread_data.fpath]
|
|
invoke con_write_asciiz, ebx
|
|
invoke con_write_asciiz, str_newline
|
|
|
|
; Start the search
|
|
invoke file.find.first, ebx, str_mask, FA_READONLY+FA_FOLDER+FA_ARCHIVED;+FA_NORMAL
|
|
test eax, eax
|
|
jz .nosuchdir
|
|
|
|
lea edi, [ebp + thread_data.buffer]
|
|
.parse_file:
|
|
test eax, eax ; did we find a file?
|
|
jz .done
|
|
mov ebx, eax ; yes, save the descripter in ebx
|
|
|
|
; first, convert the attributes
|
|
test [ebx + FileInfoA.Attributes], FA_FOLDER
|
|
jnz .folder
|
|
|
|
test [ebx + FileInfoA.Attributes], FA_READONLY
|
|
jnz .readonly
|
|
|
|
mov eax, '-rw-'
|
|
stosd
|
|
jmp .attr
|
|
|
|
.folder:
|
|
mov eax, 'drwx'
|
|
stosd
|
|
jmp .attr
|
|
|
|
.readonly:
|
|
mov eax, '-r--'
|
|
stosd
|
|
|
|
.attr:
|
|
mov eax, 'rw-r'
|
|
stosd
|
|
mov ax, 'w-'
|
|
stosw
|
|
mov al, ' '
|
|
stosb
|
|
|
|
; now..
|
|
mov ax, '1 '
|
|
stosw
|
|
|
|
; now write owner, everything is owned by FTP, woohoo!
|
|
mov eax, 'FTP '
|
|
stosd
|
|
stosd
|
|
|
|
; now the filesize in ascii
|
|
mov eax, dword[ebx + FileInfoA.FileSize]
|
|
call dword_to_ascii
|
|
mov al, ' '
|
|
stosb
|
|
|
|
; then date (month/day/year)
|
|
movzx eax, [ebx + FileInfoA.DateModify.month]
|
|
cmp eax, 12
|
|
ja @f
|
|
mov eax, [months - 4 + 4*eax]
|
|
stosd
|
|
@@:
|
|
|
|
movzx eax, [ebx + FileInfoA.DateModify.day]
|
|
call dword_to_ascii
|
|
mov al, ' '
|
|
stosb
|
|
|
|
movzx eax, [ebx + FileInfoA.DateModify.year]
|
|
call dword_to_ascii
|
|
mov al, ' '
|
|
stosb
|
|
|
|
; and last but not least, filename
|
|
lea esi, [ebx + FileInfoA.FileName]
|
|
mov ecx, 264
|
|
.nameloop:
|
|
lodsb
|
|
test al, al
|
|
jz .namedone
|
|
stosb
|
|
loop .nameloop
|
|
|
|
; insert a cr lf
|
|
.namedone:
|
|
mov ax, 0x0a0d
|
|
stosw
|
|
|
|
test [ebp + thread_data.permissions], ABORT ; Did we receive ABOR command from client?
|
|
jnz abort_transfer
|
|
|
|
; check next file
|
|
invoke file.find.next, ebx
|
|
jmp .parse_file
|
|
|
|
; close file desc
|
|
.done:
|
|
invoke file.find.close, ebx ; ebx is descriptor of last file, eax will be -1 !!
|
|
|
|
; append the string with a 0
|
|
xor al, al
|
|
stosb
|
|
|
|
; Warn the client we're about to send the data
|
|
push edi
|
|
sendFTP "150 Here it comes.."
|
|
pop esi
|
|
|
|
; and send it to the client
|
|
mov ecx, [ebp + thread_data.datasocketnum] ; socket num
|
|
lea edx, [ebp + thread_data.buffer] ; buffer ptr
|
|
sub esi, edx ; length
|
|
xor edi, edi ; flags
|
|
mcall send
|
|
|
|
; close the data socket..
|
|
mov [ebp + thread_data.mode], MODE_NOTREADY
|
|
mcall close, [ebp + thread_data.datasocketnum]
|
|
|
|
sendFTP "226 List OK"
|
|
ret
|
|
|
|
.nosuchdir:
|
|
sendFTP "550 Directory does not exist"
|
|
ret
|
|
|
|
;------------------------------------------------
|
|
; "NLST"
|
|
;
|
|
; List the filenames of the files in the current working directory.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdNLST:
|
|
|
|
test [ebp + thread_data.permissions], PERMISSION_EXEC
|
|
jz permission_denied
|
|
|
|
; TODO: same as list but simpler output format
|
|
|
|
ret
|
|
|
|
;------------------------------------------------
|
|
; "NOOP"
|
|
;
|
|
; No operation, just keep the connection alive.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdNOOP:
|
|
|
|
sendFTP "200 Command OK"
|
|
ret
|
|
|
|
;------------------------------------------------
|
|
; "PASS"
|
|
;
|
|
; Second phase of login process, client provides password.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdPASS:
|
|
|
|
; read the password from users.ini
|
|
lea edi, [ebp + thread_data.buffer + 512] ; temp pass
|
|
lea ebx, [ebp + thread_data.fpath] ; temp username
|
|
invoke ini.get_str, path2, ebx, str_pass, edi, 512, str_infinity
|
|
test eax, eax ; unable to read password? fail!
|
|
jnz .incorrect
|
|
cmp dword [edi], -1 ; no key, section or file found.. fail!
|
|
je .incorrect
|
|
cmp byte [edi], 0 ; zero password? ok!
|
|
je .ok
|
|
|
|
add esi, 5
|
|
sub ecx, 5
|
|
jbe .incorrect ; no password given? but hey, we need one! fail..
|
|
|
|
; compare with received password
|
|
repe cmpsb
|
|
cmp byte [esi-1], 0x20 ; printeable characters left?
|
|
jae .incorrect
|
|
cmp byte [edi-1], 0
|
|
jne .incorrect
|
|
|
|
.ok:
|
|
invoke ini.get_int, path2, ebx, str_mode, 0
|
|
mov [ebp + thread_data.permissions], eax
|
|
|
|
invoke con_write_asciiz, str_pass_ok
|
|
mov [ebp + thread_data.state], STATE_ACTIVE
|
|
sendFTP "230 You are now logged in"
|
|
ret
|
|
|
|
.2:
|
|
.incorrect:
|
|
invoke con_write_asciiz, str_pass_err
|
|
mov [ebp + thread_data.state], STATE_CONNECTED ; reset state
|
|
sendFTP "530 Login incorrect"
|
|
ret
|
|
|
|
.0:
|
|
sendFTP "503 Login with USER first"
|
|
ret
|
|
|
|
.3:
|
|
sendFTP "230 Already logged in"
|
|
ret
|
|
|
|
;------------------------------------------------
|
|
; "PASV"
|
|
;
|
|
; Initiate a passive dataconnection.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdPASV:
|
|
|
|
; cmp [ebp + thread_data.passivesocknum], -1
|
|
; je @f
|
|
; mcall close, [ebp + thread_data.passivesocknum] ; if there is still a socket open, close it
|
|
; @@:
|
|
|
|
; Open a new TCP socket
|
|
mcall socket, AF_INET4, SOCK_STREAM, 0
|
|
cmp eax, -1
|
|
je socketerror
|
|
mov [ebp + thread_data.passivesocknum], eax
|
|
|
|
; Bind it to a known local port
|
|
mov [ebp + thread_data.datasock.sin_family], AF_INET4
|
|
mov [ebp + thread_data.datasock.sin_addr], 0
|
|
|
|
mov ecx, eax ; passivesocketnum
|
|
lea edx, [ebp + thread_data.datasock]
|
|
mov esi, sizeof.thread_data.datasock
|
|
|
|
.next_port: ; TODO: break the endless loop
|
|
call nextpasvport
|
|
mov ax, [pasv_port]
|
|
xchg al, ah
|
|
mov [ebp + thread_data.datasock.sin_port], ax
|
|
|
|
mcall bind
|
|
cmp eax, -1
|
|
je .next_port
|
|
|
|
; And set it to listen!
|
|
mcall listen, , 1
|
|
cmp eax, -1
|
|
je socketerror
|
|
|
|
; Tell our thread we are ready to accept incoming calls
|
|
mov [ebp + thread_data.mode], MODE_PASSIVE_WAIT
|
|
|
|
; Now tell the client where to connect to in this format:
|
|
; 227 Entering Passive Mode (a1,a2,a3,a4,p1,p2)
|
|
; where a1.a2.a3.a4 is the IP address and p1*256+p2 is the port number.
|
|
|
|
; '227 Entering passive mode ('
|
|
lea edi, [ebp + thread_data.buffer]
|
|
mov ecx, str_227.length
|
|
mov esi, str_227
|
|
rep movsb
|
|
; ip
|
|
movzx eax, byte [serverip]
|
|
call dword_to_ascii
|
|
mov al, ','
|
|
stosb
|
|
movzx eax, byte [serverip+1]
|
|
call dword_to_ascii
|
|
mov al, ','
|
|
stosb
|
|
movzx eax, byte [serverip+2]
|
|
call dword_to_ascii
|
|
mov al, ','
|
|
stosb
|
|
movzx eax, byte [serverip+3]
|
|
call dword_to_ascii
|
|
mov al, ','
|
|
stosb
|
|
; port
|
|
movzx eax, byte [ebp + thread_data.datasock.sin_port]
|
|
call dword_to_ascii
|
|
mov al, ','
|
|
stosb
|
|
movzx eax, byte [ebp + thread_data.datasock.sin_port+1]
|
|
call dword_to_ascii
|
|
; ')', 13, 10, 0
|
|
mov eax, ')' + (0x000a0d shl 8)
|
|
stosd
|
|
|
|
lea esi, [edi - thread_data.buffer - 1] ; calculate length, do not cound the trailing 0 byte
|
|
sub esi, ebp
|
|
mov ecx, [ebp + thread_data.socketnum]
|
|
lea edx, [ebp + thread_data.buffer]
|
|
xor edi, edi
|
|
mcall send
|
|
|
|
invoke con_write_asciiz, edx
|
|
|
|
ret
|
|
|
|
|
|
iglobal
|
|
str_227 db "227 Entering passive mode ("
|
|
.length = $ - str_227
|
|
endg
|
|
|
|
;------------------------------------------------
|
|
; "PWD"
|
|
;
|
|
; Print the current working directory.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdPWD:
|
|
|
|
mov dword [ebp + thread_data.buffer], '257 '
|
|
mov byte [ebp + thread_data.buffer+4], '"'
|
|
|
|
lea edi, [ebp + thread_data.buffer+5]
|
|
lea esi, [ebp + thread_data.work_dir]
|
|
mov ecx, 1024
|
|
.loop:
|
|
lodsb
|
|
or al, al
|
|
jz .ok
|
|
stosb
|
|
dec ecx
|
|
jnz .loop
|
|
|
|
.ok:
|
|
mov dword [edi], '"' + 0x000a0d00 ; '"',13,10,0
|
|
lea esi, [edi - thread_data.buffer + 3]
|
|
sub esi, ebp
|
|
mov ecx, [ebp + thread_data.socketnum]
|
|
lea edx, [ebp + thread_data.buffer]
|
|
xor edi, edi
|
|
mcall send
|
|
|
|
; Print the new working dir on the console
|
|
lea eax, [ebp + thread_data.work_dir]
|
|
invoke con_write_asciiz, eax
|
|
invoke con_write_asciiz, str_newline
|
|
|
|
ret
|
|
|
|
;------------------------------------------------
|
|
; "PORT"
|
|
;
|
|
; Initiate an active dataconnection.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdPORT:
|
|
|
|
; PORT a1,a2,a3,a4,p1,p2
|
|
; IP address a1.a2.a3.a4, port p1*256+p2
|
|
|
|
; Convert the IP
|
|
lea esi, [esi+5]
|
|
mov cl, ','
|
|
call ip_to_dword
|
|
; And put it in datasock
|
|
mov [ebp + thread_data.datasock.sin_addr], ebx
|
|
|
|
; Now the same with portnumber
|
|
inc esi
|
|
call ascii_to_byte
|
|
mov byte[ebp + thread_data.datasock.sin_port], al
|
|
inc esi
|
|
call ascii_to_byte
|
|
mov byte[ebp + thread_data.datasock.sin_port+1], al
|
|
|
|
; We will open the socket, but do not connect yet!
|
|
mov [ebp + thread_data.datasock.sin_family], AF_INET4
|
|
mcall socket, AF_INET4, SOCK_STREAM, 0
|
|
cmp eax, -1
|
|
je socketerror
|
|
|
|
mov [ebp + thread_data.datasocketnum], eax
|
|
mov [ebp + thread_data.mode], MODE_ACTIVE
|
|
|
|
sendFTP "225 Data connection open"
|
|
ret
|
|
|
|
;------------------------------------------------
|
|
; "QUIT"
|
|
;
|
|
; Close the connection with client.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdQUIT:
|
|
|
|
sendFTP "221 Bye!"
|
|
mcall close, [ebp + thread_data.datasocketnum]
|
|
mcall close, [ebp + thread_data.socketnum]
|
|
|
|
add esp, 4 ; get rid of call return address
|
|
jmp thread_exit ; now close this thread
|
|
|
|
|
|
;------------------------------------------------
|
|
; "RETR"
|
|
;
|
|
; Retrieve a file from the ftp server.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdRETR:
|
|
|
|
test [ebp + thread_data.permissions], PERMISSION_READ
|
|
jz permission_denied
|
|
|
|
cmp ecx, 1024 + 5
|
|
jae .cannot_open
|
|
|
|
sub ecx, 5
|
|
jb .cannot_open
|
|
|
|
call open_datasock
|
|
|
|
call create_path
|
|
dec edi
|
|
lea esi, [ebp + thread_data.buffer + 5]
|
|
mov ecx, 1024
|
|
cmp byte [esi], '/'
|
|
jne .loop
|
|
inc esi
|
|
.loop:
|
|
lodsb
|
|
cmp al, 0x20
|
|
jl .done
|
|
stosb
|
|
loop .loop
|
|
.done:
|
|
xor al, al
|
|
stosb
|
|
|
|
lea ebx, [ebp + thread_data.fpath]
|
|
invoke con_write_asciiz, ebx
|
|
invoke con_write_asciiz, str_newline
|
|
|
|
invoke file.open, ebx, O_READ
|
|
test eax, eax
|
|
jz .cannot_open
|
|
|
|
push eax
|
|
sendFTP "150 Here it comes.."
|
|
pop ebx
|
|
|
|
.read_more:
|
|
test [ebp + thread_data.permissions], ABORT
|
|
jnz abort_transfer
|
|
|
|
lea eax, [ebp + thread_data.buffer] ; FIXME: use another buffer!! if we receive something on control connection now, we screw up!
|
|
invoke file.read, ebx, eax, BUFFERSIZE
|
|
cmp eax, -1
|
|
je .cannot_open ; FIXME: this is not the correct error
|
|
|
|
invoke con_write_asciiz, str2
|
|
|
|
push eax ebx
|
|
mov esi, eax
|
|
mov ecx, [ebp + thread_data.datasocketnum]
|
|
lea edx, [ebp + thread_data.buffer]
|
|
xor edi, edi
|
|
mcall send
|
|
pop ebx ecx
|
|
cmp eax, -1
|
|
je socketerror ; FIXME: not the correct error
|
|
|
|
; cmp eax, ecx
|
|
; jne not_all_byes_sent ; TODO
|
|
|
|
cmp ecx, BUFFERSIZE
|
|
je .read_more
|
|
|
|
invoke file.close, ebx
|
|
|
|
invoke con_write_asciiz, str2b
|
|
|
|
mov [ebp + thread_data.mode], MODE_NOTREADY
|
|
mcall close, [ebp + thread_data.datasocketnum]
|
|
|
|
sendFTP "226 Transfer OK, closing connection"
|
|
ret
|
|
|
|
.cannot_open:
|
|
invoke con_set_flags, 0x0c
|
|
invoke con_write_asciiz, str_notfound
|
|
invoke con_set_flags, 0x07
|
|
|
|
sendFTP "550 No such file"
|
|
ret
|
|
|
|
|
|
|
|
;------------------------------------------------
|
|
; "STOR"
|
|
;
|
|
; Store a file on the server.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdSTOR:
|
|
|
|
test [ebp + thread_data.permissions], PERMISSION_WRITE
|
|
jz permission_denied
|
|
|
|
;sendFTP " Ready to receive"
|
|
; open datasocket
|
|
cmp ecx, 1024 + 5
|
|
jae .cannot_open
|
|
|
|
sub ecx, 5
|
|
jb .cannot_open
|
|
|
|
call open_datasock
|
|
|
|
; creat path
|
|
call create_path
|
|
dec edi
|
|
lea esi, [ebp + thread_data.buffer + 5]
|
|
mov ecx, 1024
|
|
cmp byte [esi], '/'
|
|
jne .loop
|
|
inc esi
|
|
.loop:
|
|
lodsb
|
|
cmp al, 0x20
|
|
jl .done
|
|
stosb
|
|
loop .loop
|
|
.done:
|
|
xor al, al
|
|
stosb
|
|
|
|
lea ebx, [ebp + thread_data.fpath]
|
|
invoke con_write_asciiz, ebx
|
|
invoke con_write_asciiz, str_newline
|
|
|
|
; open file
|
|
invoke file.open, ebx, O_CREATE + O_WRITE
|
|
test eax, eax
|
|
jz .cannot_open
|
|
|
|
push eax
|
|
sendFTP "150 Here it comes.."
|
|
pop ebx
|
|
|
|
.write_more:
|
|
test [ebp + thread_data.permissions], ABORT
|
|
jnz abort_transfer
|
|
|
|
push eax ebx
|
|
mov esi, BUFFERSIZE ; eax
|
|
mov ecx, [ebp + thread_data.datasocketnum]
|
|
lea edx, [ebp + thread_data.buffer]
|
|
xor edi, edi
|
|
mcall recv
|
|
pop ebx ecx
|
|
cmp eax, -1
|
|
je socketerror ; FIXME: not the correct error
|
|
|
|
test eax, eax
|
|
jz @f
|
|
|
|
push edx
|
|
mov edx, eax
|
|
lea eax, [ebp + thread_data.buffer] ; FIXME: use another buffer!! if we receive something on control connection now, we screw up!
|
|
invoke file.write, ebx, eax, edx
|
|
pop edx
|
|
|
|
cmp eax, -1
|
|
je .cannot_open ; FIXME: this is not the correct error
|
|
|
|
invoke con_write_asciiz, str2
|
|
|
|
; cmp eax, ecx
|
|
; jne not_all_byes_sent ; TODO
|
|
|
|
;cmp ecx, BUFFERSIZE
|
|
;je .write_more
|
|
jmp .write_more
|
|
@@:
|
|
|
|
invoke file.close, ebx
|
|
|
|
invoke con_write_asciiz, str2b
|
|
|
|
mov [ebp + thread_data.mode], MODE_NOTREADY
|
|
mcall close, [ebp + thread_data.datasocketnum]
|
|
|
|
|
|
|
|
;;;; TODO
|
|
;
|
|
; test [ebp + thread_data.permissions], ABORT
|
|
; jnz abort_transfer
|
|
;
|
|
;;;;
|
|
|
|
sendFTP "226 Transfer OK"
|
|
ret
|
|
|
|
.cannot_open:
|
|
sendFTP "550 No create file"
|
|
ret
|
|
|
|
;------------------------------------------------
|
|
; "SYST"
|
|
;
|
|
; Send information about the system.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdSYST:
|
|
|
|
sendFTP "215 UNIX type: L8"
|
|
ret
|
|
|
|
;------------------------------------------------
|
|
; "TYPE"
|
|
;
|
|
; Choose the file transfer type.
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdTYPE:
|
|
|
|
cmp ecx, 6
|
|
jb parse_cmd.error
|
|
|
|
mov al, byte[esi+5]
|
|
and al, not 0x20
|
|
|
|
cmp al, 'A'
|
|
je .ascii
|
|
cmp al, 'E'
|
|
je .ebdic
|
|
cmp al, 'I'
|
|
je .image
|
|
cmp al, 'L'
|
|
je .local
|
|
|
|
jmp parse_cmd.error
|
|
|
|
.ascii:
|
|
mov [ebp + thread_data.type], TYPE_ASCII
|
|
jmp .subtype
|
|
|
|
.ebdic:
|
|
mov [ebp + thread_data.type], TYPE_EBDIC
|
|
|
|
.subtype:
|
|
cmp ecx, 8
|
|
jb .non_print
|
|
|
|
mov al, byte[esi+7]
|
|
and al, not 0x20
|
|
|
|
cmp al, 'N'
|
|
je .non_print
|
|
cmp al, 'T'
|
|
je .telnet
|
|
cmp al, 'C'
|
|
je .asacc
|
|
cmp al, 0x20
|
|
jb .non_print
|
|
|
|
jmp parse_cmd.error
|
|
|
|
.non_print:
|
|
or [ebp + thread_data.type], TYPE_NP
|
|
jmp .ok
|
|
|
|
.telnet:
|
|
or [ebp + thread_data.type], TYPE_TELNET
|
|
jmp .ok
|
|
|
|
.asacc:
|
|
or [ebp + thread_data.type], TYPE_ASA
|
|
jmp .ok
|
|
|
|
.image:
|
|
mov [ebp + thread_data.type], TYPE_IMAGE
|
|
jmp .ok
|
|
|
|
.local:
|
|
cmp ecx, 8
|
|
jb parse_cmd.error
|
|
|
|
mov al, byte[esi+7]
|
|
sub al, '0'
|
|
jb parse_cmd.error ; FIXME: this is not the correct errormessage
|
|
cmp al, 9
|
|
ja parse_cmd.error ; FIXME
|
|
or al, TYPE_LOCAL
|
|
mov [ebp + thread_data.type], al
|
|
|
|
.ok:
|
|
sendFTP "200 Command ok"
|
|
ret
|
|
|
|
;------------------------------------------------
|
|
; "USER"
|
|
;
|
|
; Login to the server, step one of two. ;;; TODO: prevent buffer overflow!
|
|
;
|
|
;------------------------------------------------
|
|
align 4
|
|
cmdUSER:
|
|
|
|
lea esi, [esi + 5]
|
|
lea edi, [ebp + thread_data.fpath] ; temp buffer for username
|
|
.loop:
|
|
lodsb
|
|
stosb
|
|
cmp al, 0x20
|
|
jae .loop
|
|
mov byte [edi-1], 0
|
|
|
|
lea esi, [ebp + thread_data.fpath]
|
|
lea eax, [ebp + thread_data.home_dir]
|
|
invoke ini.get_str, path2, esi, str_home, eax, 1024, str_infinity
|
|
cmp eax, -1
|
|
je .login_fail
|
|
cmp dword [esi], -1
|
|
je .login_fail
|
|
|
|
mov word [ebp + thread_data.work_dir], "/" ; "/", 0
|
|
|
|
invoke con_write_asciiz, str_logged_in
|
|
mov [ebp + thread_data.state], STATE_LOGIN
|
|
.sendstr:
|
|
sendFTP "331 Please specify the password"
|
|
ret
|
|
|
|
.login_fail:
|
|
invoke con_write_asciiz, str_pass_err
|
|
mov [ebp + thread_data.state], STATE_LOGIN_FAIL
|
|
jmp .sendstr
|
|
|
|
align 4
|
|
.2:
|
|
sendFTP "530 Can't change to another user"
|
|
ret
|