2016-05-07 13:42:31 +03:00
|
|
|
; dh_gex.inc - Diffie Hellman Group exchange
|
|
|
|
;
|
|
|
|
; Copyright (C) 2015-2016 Jeffrey Amelynck
|
|
|
|
;
|
|
|
|
; 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 3 of the License, or
|
|
|
|
; (at your option) any later version.
|
|
|
|
;
|
|
|
|
; 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, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
; https://www.ietf.org/rfc/rfc4419.txt
|
|
|
|
|
|
|
|
; TODO: dont convert mpints to little endian immediately.
|
|
|
|
; Or maybe even better, not at all.
|
|
|
|
|
|
|
|
proc dh_gex
|
|
|
|
|
|
|
|
;----------------------------------------------
|
|
|
|
; >> Send Diffie-Hellman Group Exchange Request
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
DEBUGF 2, "Sending GEX\n"
|
|
|
|
stdcall ssh_send_packet, con, ssh_gex_req, ssh_gex_req.length, 0
|
2016-05-07 13:42:31 +03:00
|
|
|
cmp eax, -1
|
|
|
|
je .socket_err
|
|
|
|
|
|
|
|
;---------------------------------------------
|
|
|
|
; << Parse Diffie-Hellman Group Exchange Group
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall ssh_recv_packet, con, 0
|
2016-05-07 13:42:31 +03:00
|
|
|
cmp eax, -1
|
|
|
|
je .socket_err
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
cmp [con.rx_buffer.message_code], SSH_MSG_KEX_DH_GEX_GROUP
|
2016-05-07 13:42:31 +03:00
|
|
|
jne proto_err
|
2016-08-10 19:20:49 +03:00
|
|
|
DEBUGF 2, "Received GEX group\n"
|
2016-05-07 13:42:31 +03:00
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
mov esi, con.rx_buffer+sizeof.ssh_packet_header
|
|
|
|
mov edi, con.dh_p
|
2016-05-07 13:42:31 +03:00
|
|
|
DEBUGF 1, "DH modulus (p): "
|
|
|
|
call mpint_to_little_endian
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall mpint_print, con.dh_p
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
DEBUGF 1, "DH base (g): "
|
2016-08-10 19:20:49 +03:00
|
|
|
mov edi, con.dh_g
|
2016-05-07 13:42:31 +03:00
|
|
|
call mpint_to_little_endian
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall mpint_print, con.dh_g
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;-------------------------------------------
|
|
|
|
; >> Send Diffie-Hellman Group Exchange Init
|
|
|
|
|
|
|
|
; generate a random number x, where 1 < x < (p-1)/2
|
2016-08-10 19:20:49 +03:00
|
|
|
mov edi, con.dh_x+4
|
|
|
|
mov [con.dh_x], DH_PRIVATE_KEY_SIZE/8
|
2016-05-07 13:42:31 +03:00
|
|
|
mov ecx, DH_PRIVATE_KEY_SIZE/8/4
|
|
|
|
@@:
|
|
|
|
push ecx
|
|
|
|
call MBRandom
|
|
|
|
pop ecx
|
|
|
|
stosd
|
|
|
|
dec ecx
|
|
|
|
jnz @r
|
|
|
|
|
|
|
|
; If the highest bit is set, add a zero byte
|
|
|
|
shl eax, 1
|
|
|
|
jnc @f
|
|
|
|
mov byte[edi], 0
|
2016-08-10 19:20:49 +03:00
|
|
|
inc dword[con.dh_x]
|
2016-05-07 13:42:31 +03:00
|
|
|
@@:
|
|
|
|
|
|
|
|
; Fill remaining bytes with zeros ; TO BE REMOVED ?
|
|
|
|
if ((MAX_BITS-DH_PRIVATE_KEY_SIZE) > 0)
|
|
|
|
mov ecx, (MAX_BITS-DH_PRIVATE_KEY_SIZE)/8/4
|
|
|
|
xor eax, eax
|
2016-08-10 19:20:49 +03:00
|
|
|
rep stosd
|
2016-05-07 13:42:31 +03:00
|
|
|
end if
|
|
|
|
|
|
|
|
DEBUGF 1, "DH x: "
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall mpint_print, con.dh_x
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
; Compute e = g^x mod p
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall mpint_modexp, con.dh_e, con.dh_g, con.dh_x, con.dh_p
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
DEBUGF 1, "DH e: "
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall mpint_print, con.dh_e
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
; Create group exchange init packet
|
2016-08-10 19:20:49 +03:00
|
|
|
mov edi, con.tx_buffer.message_code
|
2016-05-07 13:42:31 +03:00
|
|
|
mov al, SSH_MSG_KEX_DH_GEX_INIT
|
|
|
|
stosb
|
2016-08-10 19:20:49 +03:00
|
|
|
mov esi, con.dh_e
|
2016-05-07 13:42:31 +03:00
|
|
|
call mpint_to_big_endian
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
DEBUGF 2, "Sending GEX init\n"
|
|
|
|
mov ecx, dword[con.tx_buffer.message_code+1]
|
2016-05-07 13:42:31 +03:00
|
|
|
bswap ecx
|
|
|
|
add ecx, 5
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall ssh_send_packet, con, con.tx_buffer.message_code, ecx, 0
|
2016-05-07 13:42:31 +03:00
|
|
|
cmp eax, -1
|
|
|
|
je .socket_err
|
|
|
|
|
|
|
|
;---------------------------------------------
|
|
|
|
; << Parse Diffie-Hellman Group Exchange Reply
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall ssh_recv_packet, con, 0
|
2016-05-07 13:42:31 +03:00
|
|
|
cmp eax, -1
|
|
|
|
je .socket_err
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
cmp [con.rx_buffer.message_code], SSH_MSG_KEX_DH_GEX_REPLY
|
2016-05-07 13:42:31 +03:00
|
|
|
jne .proto_err
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
DEBUGF 2, "Received GEX Reply\n"
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;--------------------------------
|
2016-08-10 19:20:49 +03:00
|
|
|
; HASH: string K_S, the host key
|
|
|
|
mov esi, con.rx_buffer+sizeof.ssh_packet_header
|
2016-05-07 13:42:31 +03:00
|
|
|
mov edx, [esi]
|
|
|
|
bswap edx
|
|
|
|
add edx, 4
|
|
|
|
lea ebx, [esi+edx]
|
|
|
|
push ebx
|
2016-08-10 19:20:49 +03:00
|
|
|
invoke sha256_update, con.temp_ctx, esi, edx
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;--------------------------------------------------------------------------
|
2016-08-10 19:20:49 +03:00
|
|
|
; HASH: uint32 min, minimal size in bits of an acceptable group
|
|
|
|
; uint32 n, preferred size in bits of the group the server will send
|
|
|
|
; uint32 max, maximal size in bits of an acceptable group
|
|
|
|
invoke sha256_update, con.temp_ctx, ssh_gex_req+sizeof.ssh_packet_header-ssh_packet_header.message_code, 12
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;----------------------------
|
2016-08-10 19:20:49 +03:00
|
|
|
; HASH: mpint p, safe prime
|
|
|
|
mov esi, con.dh_p
|
2017-06-11 14:06:56 +03:00
|
|
|
mov edi, con.mpint_tmp
|
2016-05-07 13:42:31 +03:00
|
|
|
call mpint_to_big_endian
|
|
|
|
lea edx, [eax+4]
|
2017-06-11 14:06:56 +03:00
|
|
|
invoke sha256_update, con.temp_ctx, con.mpint_tmp, edx
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;----------------------------------------
|
2016-08-10 19:20:49 +03:00
|
|
|
; HASH: mpint g, generator for subgroup
|
|
|
|
mov esi, con.dh_g
|
2017-06-11 14:06:56 +03:00
|
|
|
mov edi, con.mpint_tmp
|
2016-05-07 13:42:31 +03:00
|
|
|
call mpint_to_big_endian
|
|
|
|
lea edx, [eax+4]
|
2017-06-11 14:06:56 +03:00
|
|
|
invoke sha256_update, con.temp_ctx, con.mpint_tmp, edx
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;---------------------------------------------------
|
2016-08-10 19:20:49 +03:00
|
|
|
; HASH: mpint e, exchange value sent by the client
|
|
|
|
mov esi, con.tx_buffer+sizeof.ssh_packet_header
|
2016-05-07 13:42:31 +03:00
|
|
|
mov edx, [esi]
|
|
|
|
bswap edx
|
|
|
|
add edx, 4
|
2016-08-10 19:20:49 +03:00
|
|
|
invoke sha256_update, con.temp_ctx, esi, edx
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;---------------------------------------------------
|
2016-08-10 19:20:49 +03:00
|
|
|
; HASH: mpint f, exchange value sent by the server
|
2016-05-07 13:42:31 +03:00
|
|
|
mov esi, [esp]
|
|
|
|
mov edx, [esi]
|
|
|
|
bswap edx
|
|
|
|
add edx, 4
|
2016-08-10 19:20:49 +03:00
|
|
|
invoke sha256_update, con.temp_ctx, esi, edx
|
2016-05-07 13:42:31 +03:00
|
|
|
pop esi
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
mov edi, con.dh_f
|
2016-05-07 13:42:31 +03:00
|
|
|
call mpint_to_little_endian
|
|
|
|
|
|
|
|
DEBUGF 1, "DH f: "
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall mpint_print, con.dh_f
|
2016-05-07 13:42:31 +03:00
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
mov edi, con.dh_signature
|
2016-05-07 13:42:31 +03:00
|
|
|
call mpint_to_little_endian
|
|
|
|
|
|
|
|
DEBUGF 1, "DH signature: "
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall mpint_print, con.dh_signature
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;--------------------------------------
|
|
|
|
; Calculate shared secret K = f^x mod p
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall mpint_modexp, con.rx_buffer, con.dh_f, con.dh_x, con.dh_p
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
DEBUGF 1, "DH K: "
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall mpint_print, con.rx_buffer
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
; We always need it in big endian order, so store it as such.
|
2016-08-10 19:20:49 +03:00
|
|
|
mov edi, con.dh_K
|
|
|
|
mov esi, con.rx_buffer
|
2016-05-07 13:42:31 +03:00
|
|
|
call mpint_to_big_endian
|
2016-08-10 19:20:49 +03:00
|
|
|
mov [con.dh_K_length], eax
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;-----------------------------------
|
2016-08-10 19:20:49 +03:00
|
|
|
; HASH: mpint K, the shared secret
|
|
|
|
mov edx, [con.dh_K_length]
|
2016-05-07 13:42:31 +03:00
|
|
|
add edx, 4
|
2016-08-10 19:20:49 +03:00
|
|
|
invoke sha256_update, con.temp_ctx, con.dh_K, edx
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;-------------------------------
|
|
|
|
; Finalize the exchange hash (H)
|
2016-08-10 19:20:49 +03:00
|
|
|
invoke sha256_final, con.temp_ctx
|
|
|
|
mov esi, con.temp_ctx.hash
|
|
|
|
mov edi, con.dh_H
|
|
|
|
mov ecx, SHA256_HASH_SIZE/4
|
|
|
|
rep movsd
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
DEBUGF 1, "Exchange hash H: "
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall dump_hex, con.dh_H, 8
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
; TODO: skip this block when re-keying
|
2016-08-10 19:20:49 +03:00
|
|
|
mov esi, con.dh_H
|
|
|
|
mov edi, con.session_id
|
|
|
|
mov ecx, SHA256_HASH_SIZE/4
|
2016-05-07 13:42:31 +03:00
|
|
|
rep movsd
|
|
|
|
|
|
|
|
;---------------
|
|
|
|
; Calculate keys
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
; First, calculate partial hash of K and H so we can re-use it for every key.
|
|
|
|
|
|
|
|
invoke sha256_init, con.k_h_ctx
|
|
|
|
|
|
|
|
mov edx, [con.dh_K_length]
|
|
|
|
add edx, 4
|
|
|
|
invoke sha256_update, con.k_h_ctx, con.dh_K, edx
|
|
|
|
invoke sha256_update, con.k_h_ctx, con.dh_H, 32
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;---------------------------------------------------------------
|
|
|
|
; Initial IV client to server: HASH(K || H || "A" || session_id)
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
mov esi, con.k_h_ctx
|
|
|
|
mov edi, con.temp_ctx
|
2019-10-22 01:33:41 +03:00
|
|
|
mov ecx, sizeof.crash_ctx
|
2016-08-10 19:20:49 +03:00
|
|
|
rep movsd
|
|
|
|
mov [con.session_id_prefix], 'A'
|
|
|
|
invoke sha256_update, con.temp_ctx, con.session_id_prefix, 32+1
|
|
|
|
invoke sha256_final, con.temp_ctx.hash
|
|
|
|
mov edi, con.tx_iv
|
|
|
|
mov esi, con.temp_ctx
|
|
|
|
mov ecx, SHA256_HASH_SIZE/4
|
|
|
|
rep movsd
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
DEBUGF 1, "Remote IV: "
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall dump_hex, con.tx_iv, 8
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;---------------------------------------------------------------
|
|
|
|
; Initial IV server to client: HASH(K || H || "B" || session_id)
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
mov esi, con.k_h_ctx
|
|
|
|
mov edi, con.temp_ctx
|
2019-10-22 01:33:41 +03:00
|
|
|
mov ecx, sizeof.crash_ctx/4
|
2016-08-10 19:20:49 +03:00
|
|
|
rep movsd
|
|
|
|
inc [con.session_id_prefix]
|
|
|
|
invoke sha256_update, con.temp_ctx, con.session_id_prefix, 32+1
|
|
|
|
invoke sha256_final, con.temp_ctx
|
|
|
|
mov edi, con.rx_iv
|
|
|
|
mov esi, con.temp_ctx
|
|
|
|
mov ecx, SHA256_HASH_SIZE/4
|
|
|
|
rep movsd
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
DEBUGF 1, "Local IV: "
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall dump_hex, con.rx_iv, 8
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;-------------------------------------------------------------------
|
|
|
|
; Encryption key client to server: HASH(K || H || "C" || session_id)
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
mov esi, con.k_h_ctx
|
|
|
|
mov edi, con.temp_ctx
|
2019-10-22 01:33:41 +03:00
|
|
|
mov ecx, sizeof.crash_ctx
|
2016-08-10 19:20:49 +03:00
|
|
|
rep movsd
|
|
|
|
inc [con.session_id_prefix]
|
|
|
|
invoke sha256_update, con.temp_ctx, con.session_id_prefix, 32+1
|
|
|
|
invoke sha256_final, con.temp_ctx
|
|
|
|
mov edi, con.tx_enc_key
|
|
|
|
mov esi, con.temp_ctx
|
|
|
|
mov ecx, SHA256_HASH_SIZE/4
|
|
|
|
rep movsd
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
DEBUGF 1, "Remote key: "
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall dump_hex, con.tx_enc_key, 8
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;-------------------------------------------------------------------
|
|
|
|
; Encryption key server to client: HASH(K || H || "D" || session_id)
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
mov esi, con.k_h_ctx
|
|
|
|
mov edi, con.temp_ctx
|
2019-10-22 01:33:41 +03:00
|
|
|
mov ecx, sizeof.crash_ctx/4
|
2016-08-10 19:20:49 +03:00
|
|
|
rep movsd
|
|
|
|
inc [con.session_id_prefix]
|
|
|
|
invoke sha256_update, con.temp_ctx, con.session_id_prefix, 32+1
|
|
|
|
invoke sha256_final, con.temp_ctx
|
|
|
|
mov edi, con.rx_enc_key
|
|
|
|
mov esi, con.temp_ctx
|
|
|
|
mov ecx, SHA256_HASH_SIZE/4
|
|
|
|
rep movsd
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
DEBUGF 1, "Local key: "
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall dump_hex, con.rx_enc_key, 8
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;------------------------------------------------------------------
|
|
|
|
; Integrity key client to server: HASH(K || H || "E" || session_id)
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
mov esi, con.k_h_ctx
|
|
|
|
mov edi, con.temp_ctx
|
2019-10-22 01:33:41 +03:00
|
|
|
mov ecx, sizeof.crash_ctx/4
|
2016-08-10 19:20:49 +03:00
|
|
|
rep movsd
|
|
|
|
inc [con.session_id_prefix]
|
|
|
|
invoke sha256_update, con.temp_ctx, con.session_id_prefix, 32+1
|
|
|
|
invoke sha256_final, con.temp_ctx
|
|
|
|
mov edi, con.tx_int_key
|
|
|
|
mov esi, con.temp_ctx
|
|
|
|
mov ecx, SHA256_HASH_SIZE/4
|
|
|
|
rep movsd
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
DEBUGF 1, "Remote Integrity key: "
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall dump_hex, con.tx_int_key, 8
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;------------------------------------------------------------------
|
|
|
|
; Integrity key server to client: HASH(K || H || "F" || session_id)
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
mov esi, con.k_h_ctx
|
|
|
|
mov edi, con.temp_ctx
|
2019-10-22 01:33:41 +03:00
|
|
|
mov ecx, sizeof.crash_ctx/4
|
2016-08-10 19:20:49 +03:00
|
|
|
rep movsd
|
|
|
|
inc [con.session_id_prefix]
|
|
|
|
invoke sha256_update, con.temp_ctx, con.session_id_prefix, 32+1
|
|
|
|
invoke sha256_final, con.temp_ctx
|
|
|
|
mov edi, con.rx_int_key
|
|
|
|
mov esi, con.temp_ctx
|
|
|
|
mov ecx, SHA256_HASH_SIZE/4
|
|
|
|
rep movsd
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
DEBUGF 1, "Local Integrity key: "
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall dump_hex, con.rx_int_key, 8
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;-------------------------------------
|
|
|
|
; << Parse Diffie-Hellman New Keys MSG
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall ssh_recv_packet, con, 0
|
2016-05-07 13:42:31 +03:00
|
|
|
cmp eax, -1
|
|
|
|
je .socket_err
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
cmp [con.rx_buffer.message_code], SSH_MSG_NEWKEYS
|
2016-05-07 13:42:31 +03:00
|
|
|
jne .proto_err
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
DEBUGF 2, "Received New Keys\n"
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
;-------------------------------
|
|
|
|
; >> Reply with New Keys message
|
|
|
|
|
2016-08-10 19:20:49 +03:00
|
|
|
stdcall ssh_send_packet, con, ssh_new_keys, ssh_new_keys.length, 0
|
2016-05-07 13:42:31 +03:00
|
|
|
|
|
|
|
xor eax, eax
|
|
|
|
ret
|
|
|
|
|
|
|
|
.socket_err:
|
2016-08-10 19:20:49 +03:00
|
|
|
DEBUGF 3, "Socket error during key exchange!\n"
|
2016-05-07 13:42:31 +03:00
|
|
|
mov eax, 1
|
|
|
|
ret
|
|
|
|
|
|
|
|
.proto_err:
|
2016-08-10 19:20:49 +03:00
|
|
|
DEBUGF 3, "Protocol error during key exchange!\n"
|
2016-05-07 13:42:31 +03:00
|
|
|
mov eax, 2
|
|
|
|
ret
|
|
|
|
|
|
|
|
endp
|