2014-03-15 16:59:03 +00:00
// SONIC ROBO BLAST 2
//-----------------------------------------------------------------------------
// Copyright (C) 1998-2000 by DooM Legacy Team.
2018-11-25 12:35:38 +00:00
// Copyright (C) 1999-2018 by Sonic Team Junior.
2014-03-15 16:59:03 +00:00
//
// This program is free software distributed under the
// terms of the GNU General Public License, version 2.
// See the 'LICENSE' file for more details.
//-----------------------------------------------------------------------------
/// \file d_netfil.c
/// \brief Transfer a file using HSendPacket.
# include <stdio.h>
# ifndef _WIN32_WCE
# ifdef __OS2__
# include <sys/types.h>
# endif // __OS2__
# include <sys/stat.h>
# endif
# if !defined (UNDER_CE)
# include <time.h>
# endif
# if ((defined (_WIN32) && !defined (_WIN32_WCE)) || defined (__DJGPP__)) && !defined (_XBOX)
# include <io.h>
# include <direct.h>
# elif !defined (_WIN32_WCE) && !(defined (_XBOX) && !defined (__GNUC__))
# include <sys/types.h>
# include <dirent.h>
# include <utime.h>
# endif
# ifdef __GNUC__
# include <unistd.h>
# include <limits.h>
# elif defined (_WIN32) && !defined (_WIN32_WCE)
# include <sys/utime.h>
# endif
# ifdef __DJGPP__
# include <dir.h>
# include <utime.h>
# endif
# include "doomdef.h"
# include "doomstat.h"
# include "d_main.h"
# include "g_game.h"
# include "i_net.h"
# include "i_system.h"
# include "m_argv.h"
# include "d_net.h"
# include "w_wad.h"
# include "d_netfil.h"
# include "z_zone.h"
# include "byteptr.h"
# include "p_setup.h"
# include "m_misc.h"
# include "m_menu.h"
# include "md5.h"
# include "filesrch.h"
2014-04-14 05:14:58 +00:00
# include <errno.h>
2017-05-26 12:39:54 +00:00
// Prototypes
static boolean SV_SendFile ( INT32 node , const char * filename , UINT8 fileid ) ;
2014-03-15 16:59:03 +00:00
2016-12-31 18:26:33 +00:00
// Sender structure
2014-03-15 16:59:03 +00:00
typedef struct filetx_s
{
INT32 ram ;
2016-12-31 18:26:33 +00:00
union {
char * filename ; // Name of the file
char * ram ; // Pointer to the data in RAM
} id ;
UINT32 size ; // Size of the file
2014-03-15 16:59:03 +00:00
UINT8 fileid ;
2016-12-31 18:26:33 +00:00
INT32 node ; // Destination
struct filetx_s * next ; // Next file in the list
2014-03-15 16:59:03 +00:00
} filetx_t ;
2016-12-31 18:26:33 +00:00
// Current transfers (one for each node)
2014-03-15 16:59:03 +00:00
typedef struct filetran_s
{
2016-12-31 18:26:33 +00:00
filetx_t * txlist ; // Linked list of all files for the node
UINT32 position ; // The current position in the file
FILE * currentfile ; // The file currently being sent/received
2014-03-15 16:59:03 +00:00
} filetran_t ;
static filetran_t transfer [ MAXNETNODES ] ;
2016-12-31 18:26:33 +00:00
// Read time of file: stat _stmtime
// Write time of file: utime
2014-03-15 16:59:03 +00:00
2016-12-31 18:26:33 +00:00
// Receiver structure
INT32 fileneedednum ; // Number of files needed to join the server
fileneeded_t fileneeded [ MAX_WADFILES ] ; // List of needed files
2018-11-14 21:50:52 +00:00
char downloaddir [ 512 ] = " DOWNLOAD " ;
2014-03-15 16:59:03 +00:00
# ifdef CLIENT_LOADINGSCREEN
// for cl loading screen
2017-01-13 19:53:52 +00:00
INT32 lastfilenum = - 1 ;
2014-03-15 16:59:03 +00:00
# endif
/** Fills a serverinfo packet with information about wad files loaded.
*
* \ todo Give this function a better name since it is in global scope .
2018-08-28 20:08:47 +00:00
* Used to have size limiting built in - now handed via W_LoadWadFile in w_wad . c
2016-12-31 18:26:33 +00:00
*
2014-03-15 16:59:03 +00:00
*/
2019-04-18 04:51:15 +00:00
UINT8 * PutFileNeeded ( UINT16 firstfile )
2014-03-15 16:59:03 +00:00
{
2019-04-18 04:29:27 +00:00
size_t i ;
UINT8 count = 0 ;
2019-04-18 04:51:15 +00:00
UINT8 * p_start = netbuffer - > u . serverinfo . fileneeded ;
UINT8 * p = p_start ;
2014-03-15 16:59:03 +00:00
char wadfilename [ MAX_WADPATH ] = " " ;
UINT8 filestatus ;
2019-04-18 04:51:15 +00:00
for ( i = mainwads ; i < numwadfiles ; i + + )
2014-03-15 16:59:03 +00:00
{
2018-08-28 20:08:47 +00:00
// If it has only music/sound lumps, don't put it in the list
if ( ! wadfiles [ i ] - > important )
continue ;
2019-04-18 04:51:15 +00:00
if ( firstfile )
{ // Skip files until we reach the first file.
firstfile - - ;
continue ;
}
2019-04-18 04:29:27 +00:00
nameonly ( strcpy ( wadfilename , wadfiles [ i ] - > filename ) ) ;
2019-04-18 04:51:15 +00:00
if ( p + 1 + 4 + strlen ( wadfilename ) + 16 > p_start + MAXFILENEEDED )
2019-04-18 04:29:27 +00:00
{
2019-04-18 04:51:15 +00:00
// Too many files to send all at once
netbuffer - > u . serverinfo . kartvars | = SV_LOTSOFADDONS ;
2019-04-18 04:29:27 +00:00
break ;
}
2018-08-28 20:08:47 +00:00
filestatus = 1 ; // Importance - not really used any more, holds 1 by default for backwards compat with MS
2014-03-15 16:59:03 +00:00
// Store in the upper four bits
if ( ! cv_downloading . value )
2016-12-31 18:26:33 +00:00
filestatus + = ( 2 < < 4 ) ; // Won't send
2018-08-28 20:08:47 +00:00
else if ( ( wadfiles [ i ] - > filesize < = ( UINT32 ) cv_maxsend . value * 1024 ) )
2016-12-31 18:26:33 +00:00
filestatus + = ( 1 < < 4 ) ; // Will send if requested
2018-08-28 20:08:47 +00:00
// else
// filestatus += (0 << 4); -- Won't send, too big
2014-03-15 16:59:03 +00:00
WRITEUINT8 ( p , filestatus ) ;
count + + ;
WRITEUINT32 ( p , wadfiles [ i ] - > filesize ) ;
WRITESTRINGN ( p , wadfilename , MAX_WADPATH ) ;
WRITEMEM ( p , wadfiles [ i ] - > md5sum , 16 ) ;
}
2019-04-18 04:29:27 +00:00
netbuffer - > u . serverinfo . fileneedednum = count ;
2014-03-15 16:59:03 +00:00
return p ;
}
2016-12-31 18:26:33 +00:00
/** Parses the serverinfo packet and fills the fileneeded table on client
*
2019-04-18 04:42:49 +00:00
* \ param fileneedednum_parm The number of files ( sent in this page ) needed to join the server
2016-12-31 18:26:33 +00:00
* \ param fileneededstr The memory block containing the list of needed files
2019-04-18 04:42:49 +00:00
* \ param firstfile The first file index to read from
2016-12-31 18:26:33 +00:00
*/
2019-04-18 04:42:49 +00:00
void D_ParseFileneeded ( INT32 fileneedednum_parm , UINT8 * fileneededstr , UINT16 firstfile )
2014-03-15 16:59:03 +00:00
{
INT32 i ;
UINT8 * p ;
UINT8 filestatus ;
2019-04-18 04:51:15 +00:00
fileneedednum = firstfile + fileneedednum_parm ;
2014-03-15 16:59:03 +00:00
p = ( UINT8 * ) fileneededstr ;
2019-04-18 04:42:49 +00:00
for ( i = firstfile ; i < fileneedednum ; i + + )
2014-03-15 16:59:03 +00:00
{
2016-12-31 18:26:33 +00:00
fileneeded [ i ] . status = FS_NOTFOUND ; // We haven't even started looking for the file yet
filestatus = READUINT8 ( p ) ; // The first byte is the file status
2014-03-15 16:59:03 +00:00
fileneeded [ i ] . willsend = ( UINT8 ) ( filestatus > > 4 ) ;
2016-12-31 18:26:33 +00:00
fileneeded [ i ] . totalsize = READUINT32 ( p ) ; // The four next bytes are the file size
fileneeded [ i ] . file = NULL ; // The file isn't open yet
READSTRINGN ( p , fileneeded [ i ] . filename , MAX_WADPATH ) ; // The next bytes are the file name
READMEM ( p , fileneeded [ i ] . md5sum , 16 ) ; // The last 16 bytes are the file checksum
2014-03-15 16:59:03 +00:00
}
}
void CL_PrepareDownloadSaveGame ( const char * tmpsave )
{
fileneedednum = 1 ;
fileneeded [ 0 ] . status = FS_REQUESTED ;
fileneeded [ 0 ] . totalsize = UINT32_MAX ;
2016-12-31 18:26:33 +00:00
fileneeded [ 0 ] . file = NULL ;
2014-03-15 16:59:03 +00:00
memset ( fileneeded [ 0 ] . md5sum , 0 , 16 ) ;
strcpy ( fileneeded [ 0 ] . filename , tmpsave ) ;
}
/** Checks the server to see if we CAN download all the files,
* before starting to create them and requesting .
2016-12-31 18:26:33 +00:00
*
* \ return True if we can download all the files
*
2014-03-15 16:59:03 +00:00
*/
boolean CL_CheckDownloadable ( void )
{
UINT8 i , dlstatus = 0 ;
for ( i = 0 ; i < fileneedednum ; i + + )
2018-08-28 20:08:47 +00:00
if ( fileneeded [ i ] . status ! = FS_FOUND & & fileneeded [ i ] . status ! = FS_OPEN )
2014-03-15 16:59:03 +00:00
{
if ( fileneeded [ i ] . willsend = = 1 )
continue ;
if ( fileneeded [ i ] . willsend = = 0 )
dlstatus = 1 ;
else //if (fileneeded[i].willsend == 2)
dlstatus = 2 ;
}
// Downloading locally disabled
if ( ! dlstatus & & M_CheckParm ( " -nodownload " ) )
dlstatus = 3 ;
if ( ! dlstatus )
return true ;
// not downloadable, put reason in console
CONS_Alert ( CONS_NOTICE , M_GetText ( " You need additional files to connect to this server: \n " ) ) ;
for ( i = 0 ; i < fileneedednum ; i + + )
2018-08-28 20:08:47 +00:00
if ( fileneeded [ i ] . status ! = FS_FOUND & & fileneeded [ i ] . status ! = FS_OPEN )
2014-03-15 16:59:03 +00:00
{
CONS_Printf ( " * \" %s \" (%dK) " , fileneeded [ i ] . filename , fileneeded [ i ] . totalsize > > 10 ) ;
if ( fileneeded [ i ] . status = = FS_NOTFOUND )
CONS_Printf ( M_GetText ( " not found, md5: " ) ) ;
else if ( fileneeded [ i ] . status = = FS_MD5SUMBAD )
CONS_Printf ( M_GetText ( " wrong version, md5: " ) ) ;
{
INT32 j ;
char md5tmp [ 33 ] ;
for ( j = 0 ; j < 16 ; j + + )
sprintf ( & md5tmp [ j * 2 ] , " %02x " , fileneeded [ i ] . md5sum [ j ] ) ;
CONS_Printf ( " %s " , md5tmp ) ;
}
CONS_Printf ( " \n " ) ;
}
switch ( dlstatus )
{
case 1 :
CONS_Printf ( M_GetText ( " Some files are larger than the server is willing to send. \n " ) ) ;
break ;
case 2 :
CONS_Printf ( M_GetText ( " The server is not allowing download requests. \n " ) ) ;
break ;
case 3 :
CONS_Printf ( M_GetText ( " All files downloadable, but you have chosen to disable downloading locally. \n " ) ) ;
break ;
}
return false ;
}
2016-12-31 18:26:33 +00:00
/** Sends requests for files in the ::fileneeded table with a status of
2014-03-15 16:59:03 +00:00
* : : FS_NOTFOUND .
2016-12-31 18:26:33 +00:00
*
* \ return True if the packet was successfully sent
* \ note Sends a PT_REQUESTFILE packet
*
2014-03-15 16:59:03 +00:00
*/
boolean CL_SendRequestFile ( void )
{
char * p ;
INT32 i ;
INT64 totalfreespaceneeded = 0 , availablefreespace ;
# ifdef PARANOIA
if ( M_CheckParm ( " -nodownload " ) )
I_Error ( " Attempted to download files in -nodownload mode " ) ;
for ( i = 0 ; i < fileneedednum ; i + + )
if ( fileneeded [ i ] . status ! = FS_FOUND & & fileneeded [ i ] . status ! = FS_OPEN
2018-08-28 20:08:47 +00:00
& & ( fileneeded [ i ] . willsend = = 0 | | fileneeded [ i ] . willsend = = 2 ) )
2014-03-15 16:59:03 +00:00
{
I_Error ( " Attempted to download files that were not sendable " ) ;
}
# endif
netbuffer - > packettype = PT_REQUESTFILE ;
p = ( char * ) netbuffer - > u . textcmd ;
for ( i = 0 ; i < fileneedednum ; i + + )
2018-08-28 20:08:47 +00:00
if ( ( fileneeded [ i ] . status = = FS_NOTFOUND | | fileneeded [ i ] . status = = FS_MD5SUMBAD ) )
2014-03-15 16:59:03 +00:00
{
totalfreespaceneeded + = fileneeded [ i ] . totalsize ;
nameonly ( fileneeded [ i ] . filename ) ;
WRITEUINT8 ( p , i ) ; // fileid
WRITESTRINGN ( p , fileneeded [ i ] . filename , MAX_WADPATH ) ;
// put it in download dir
strcatbf ( fileneeded [ i ] . filename , downloaddir , " / " ) ;
fileneeded [ i ] . status = FS_REQUESTED ;
}
WRITEUINT8 ( p , 0xFF ) ;
I_GetDiskFreeSpace ( & availablefreespace ) ;
if ( totalfreespaceneeded > availablefreespace )
I_Error ( " To play on this server you must download %s KB, \n "
" but you have only %s KB free space on this drive \n " ,
sizeu1 ( ( size_t ) ( totalfreespaceneeded > > 10 ) ) , sizeu2 ( ( size_t ) ( availablefreespace > > 10 ) ) ) ;
// prepare to download
I_mkdir ( downloaddir , 0755 ) ;
return HSendPacket ( servernode , true , 0 , p - ( char * ) netbuffer - > u . textcmd ) ;
}
// get request filepak and put it on the send queue
2017-05-26 12:39:54 +00:00
// returns false if a requested file was not found or cannot be sent
boolean Got_RequestFilePak ( INT32 node )
2014-03-15 16:59:03 +00:00
{
char wad [ MAX_WADPATH + 1 ] ;
UINT8 * p = netbuffer - > u . textcmd ;
UINT8 id ;
while ( p < netbuffer - > u . textcmd + MAXTEXTCMD - 1 ) // Don't allow hacked client to overflow
{
id = READUINT8 ( p ) ;
if ( id = = 0xFF )
break ;
READSTRINGN ( p , wad , MAX_WADPATH ) ;
2017-05-26 12:39:54 +00:00
if ( ! SV_SendFile ( node , wad , id ) )
{
2017-05-26 13:19:18 +00:00
SV_AbortSendFiles ( node ) ;
2017-05-26 12:39:54 +00:00
return false ; // don't read the rest of the files
}
2014-03-15 16:59:03 +00:00
}
2017-05-26 12:39:54 +00:00
return true ; // no problems with any files
2014-03-15 16:59:03 +00:00
}
2016-12-31 18:26:33 +00:00
/** Checks if the files needed aren't already loaded or on the disk
*
* \ return 0 if some files are missing
* 1 if all files exist
* 2 if some already loaded files are not requested or are in a different order
*
*/
2014-03-15 16:59:03 +00:00
INT32 CL_CheckFiles ( void )
{
INT32 i , j ;
char wadfilename [ MAX_WADPATH ] ;
INT32 ret = 1 ;
2017-05-25 15:06:39 +00:00
size_t packetsize = 0 ;
2017-05-26 15:16:10 +00:00
size_t filestoget = 0 ;
2014-03-15 16:59:03 +00:00
// if (M_CheckParm("-nofiles"))
// return 1;
// the first is the iwad (the main wad file)
// we don't care if it's called srb2.srb or srb2.wad.
// Never download the IWAD, just assume it's there and identical
fileneeded [ 0 ] . status = FS_OPEN ;
// Modified game handling -- check for an identical file list
// must be identical in files loaded AND in order
// Return 2 on failure -- disconnect from server
if ( modifiedgame )
{
CONS_Debug ( DBG_NETPLAY , " game is modified; only doing basic checks \n " ) ;
for ( i = 1 , j = 1 ; i < fileneedednum | | j < numwadfiles ; )
{
2018-08-28 20:08:47 +00:00
if ( j < numwadfiles & & ! wadfiles [ j ] - > important )
2014-03-15 16:59:03 +00:00
{
2016-12-31 18:26:33 +00:00
// Unimportant on our side. still don't care.
2014-03-15 16:59:03 +00:00
+ + j ;
continue ;
}
// If this test is true, we've reached the end of one file list
// and the other still has a file that's important
if ( i > = fileneedednum | | j > = numwadfiles )
return 2 ;
2016-12-31 18:26:33 +00:00
// For the sake of speed, only bother with a md5 check
2014-03-15 16:59:03 +00:00
if ( memcmp ( wadfiles [ j ] - > md5sum , fileneeded [ i ] . md5sum , 16 ) )
return 2 ;
2016-12-31 18:26:33 +00:00
// It's accounted for! let's keep going.
2014-03-15 16:59:03 +00:00
CONS_Debug ( DBG_NETPLAY , " '%s' accounted for \n " , fileneeded [ i ] . filename ) ;
fileneeded [ i ] . status = FS_OPEN ;
+ + i ;
+ + j ;
}
return 1 ;
}
for ( i = 1 ; i < fileneedednum ; i + + )
{
CONS_Debug ( DBG_NETPLAY , " searching for '%s' " , fileneeded [ i ] . filename ) ;
2016-12-31 18:26:33 +00:00
// Check in already loaded files
2014-03-15 16:59:03 +00:00
for ( j = 1 ; wadfiles [ j ] ; j + + )
{
nameonly ( strcpy ( wadfilename , wadfiles [ j ] - > filename ) ) ;
if ( ! stricmp ( wadfilename , fileneeded [ i ] . filename ) & &
! memcmp ( wadfiles [ j ] - > md5sum , fileneeded [ i ] . md5sum , 16 ) )
{
CONS_Debug ( DBG_NETPLAY , " already loaded \n " ) ;
fileneeded [ i ] . status = FS_OPEN ;
break ;
}
}
2018-08-28 20:08:47 +00:00
if ( fileneeded [ i ] . status ! = FS_NOTFOUND )
2014-03-15 16:59:03 +00:00
continue ;
2017-05-25 15:06:39 +00:00
packetsize + = nameonlylength ( fileneeded [ i ] . filename ) + 22 ;
2017-05-26 15:16:10 +00:00
if ( ( numwadfiles + filestoget > = MAX_WADFILES )
2018-08-28 20:08:47 +00:00
| | ( packetsize > MAXFILENEEDED * sizeof ( UINT8 ) ) )
2017-05-25 15:06:39 +00:00
return 3 ;
2017-05-26 15:16:10 +00:00
filestoget + + ;
2014-03-15 16:59:03 +00:00
fileneeded [ i ] . status = findfile ( fileneeded [ i ] . filename , fileneeded [ i ] . md5sum , true ) ;
CONS_Debug ( DBG_NETPLAY , " found %d \n " , fileneeded [ i ] . status ) ;
if ( fileneeded [ i ] . status ! = FS_FOUND )
ret = 0 ;
}
return ret ;
}
2016-12-31 18:26:33 +00:00
// Load it now
2014-03-15 16:59:03 +00:00
void CL_LoadServerFiles ( void )
{
INT32 i ;
// if (M_CheckParm("-nofiles"))
// return;
for ( i = 1 ; i < fileneedednum ; i + + )
{
if ( fileneeded [ i ] . status = = FS_OPEN )
2016-12-31 18:26:33 +00:00
continue ; // Already loaded
2014-03-15 16:59:03 +00:00
else if ( fileneeded [ i ] . status = = FS_FOUND )
{
2018-11-23 15:58:16 +00:00
P_AddWadFile ( fileneeded [ i ] . filename ) ;
2019-01-17 22:01:28 +00:00
G_SetGameModified ( true , false ) ;
2014-03-15 16:59:03 +00:00
fileneeded [ i ] . status = FS_OPEN ;
}
else if ( fileneeded [ i ] . status = = FS_MD5SUMBAD )
2018-11-29 15:07:35 +00:00
I_Error ( " Wrong version of file %s " , fileneeded [ i ] . filename ) ;
else
2016-12-31 18:26:33 +00:00
{
2017-01-01 22:27:06 +00:00
const char * s ;
2016-12-31 18:26:33 +00:00
switch ( fileneeded [ i ] . status )
{
case FS_NOTFOUND :
s = " FS_NOTFOUND " ;
break ;
case FS_REQUESTED :
s = " FS_REQUESTED " ;
break ;
case FS_DOWNLOADING :
s = " FS_DOWNLOADING " ;
break ;
default :
s = " unknown " ;
break ;
}
I_Error ( " Try to load file \" %s \" with status of %d (%s) \n " , fileneeded [ i ] . filename ,
fileneeded [ i ] . status , s ) ;
}
2014-03-15 16:59:03 +00:00
}
}
2016-12-31 18:26:33 +00:00
// Number of files to send
// Little optimization to quickly test if there is a file in the queue
static INT32 filestosend = 0 ;
2014-03-15 16:59:03 +00:00
2016-12-31 18:26:33 +00:00
/** Adds a file to the file list for a node
*
* \ param node The node to send the file to
* \ param filename The file to send
* \ param fileid ? ? ?
* \ sa SV_SendRam
*
*/
2017-05-26 12:39:54 +00:00
static boolean SV_SendFile ( INT32 node , const char * filename , UINT8 fileid )
2014-03-15 16:59:03 +00:00
{
2016-12-31 18:26:33 +00:00
filetx_t * * q ; // A pointer to the "next" field of the last file in the list
filetx_t * p ; // The new file request
2014-03-15 16:59:03 +00:00
INT32 i ;
char wadfilename [ MAX_WADPATH ] ;
2017-01-13 19:53:52 +00:00
if ( cv_noticedownload . value )
2017-05-26 13:38:59 +00:00
CONS_Printf ( " Sending file \" %s \" to node %d (%s) \n " , filename , node , I_GetNodeAddress ( node ) ) ;
2017-01-13 19:53:52 +00:00
2016-12-31 18:26:33 +00:00
// Find the last file in the list and set a pointer to its "next" field
2014-03-15 16:59:03 +00:00
q = & transfer [ node ] . txlist ;
while ( * q )
q = & ( ( * q ) - > next ) ;
2016-12-31 18:26:33 +00:00
// Allocate a file request and append it to the file list
2014-03-15 16:59:03 +00:00
p = * q = ( filetx_t * ) malloc ( sizeof ( filetx_t ) ) ;
2016-12-31 18:26:33 +00:00
if ( ! p )
I_Error ( " SV_SendFile: No more memory \n " ) ;
// Initialise with zeros
memset ( p , 0 , sizeof ( filetx_t ) ) ;
// Allocate the file name
p - > id . filename = ( char * ) malloc ( MAX_WADPATH ) ;
if ( ! p - > id . filename )
I_Error ( " SV_SendFile: No more memory \n " ) ;
2014-03-15 16:59:03 +00:00
2016-12-31 18:26:33 +00:00
// Set the file name and get rid of the path
strlcpy ( p - > id . filename , filename , MAX_WADPATH ) ;
nameonly ( p - > id . filename ) ;
2014-03-15 16:59:03 +00:00
2016-12-31 18:26:33 +00:00
// Look for the requested file through all loaded files
2014-03-15 16:59:03 +00:00
for ( i = 0 ; wadfiles [ i ] ; i + + )
{
strlcpy ( wadfilename , wadfiles [ i ] - > filename , MAX_WADPATH ) ;
nameonly ( wadfilename ) ;
2016-12-31 18:26:33 +00:00
if ( ! stricmp ( wadfilename , p - > id . filename ) )
2014-03-15 16:59:03 +00:00
{
2016-12-31 18:26:33 +00:00
// Copy file name with full path
strlcpy ( p - > id . filename , wadfiles [ i ] - > filename , MAX_WADPATH ) ;
2014-03-15 16:59:03 +00:00
break ;
}
}
2016-12-31 18:26:33 +00:00
// Handle non-loaded file requests
2014-03-15 16:59:03 +00:00
if ( ! wadfiles [ i ] )
{
DEBFILE ( va ( " %s not found in wadfiles \n " , filename ) ) ;
2016-12-31 18:26:33 +00:00
// This formerly checked if (!findfile(p->id.filename, NULL, true))
2014-03-15 16:59:03 +00:00
2016-12-31 18:26:33 +00:00
// Not found
// Don't inform client (probably someone who thought they could leak 2.2 ACZ)
2014-03-15 16:59:03 +00:00
DEBFILE ( va ( " Client %d request %s: not found \n " , node , filename ) ) ;
2016-12-31 18:26:33 +00:00
free ( p - > id . filename ) ;
2014-03-15 16:59:03 +00:00
free ( p ) ;
* q = NULL ;
2017-05-26 12:39:54 +00:00
return false ; // cancel the rest of the requests
2014-03-15 16:59:03 +00:00
}
2016-12-31 18:26:33 +00:00
// Handle huge file requests (i.e. bigger than cv_maxsend.value KB)
2014-03-15 16:59:03 +00:00
if ( wadfiles [ i ] - > filesize > ( UINT32 ) cv_maxsend . value * 1024 )
{
2016-12-31 18:26:33 +00:00
// Too big
// Don't inform client (client sucks, man)
2014-03-15 16:59:03 +00:00
DEBFILE ( va ( " Client %d request %s: file too big, not sending \n " , node , filename ) ) ;
2016-12-31 18:26:33 +00:00
free ( p - > id . filename ) ;
2014-03-15 16:59:03 +00:00
free ( p ) ;
* q = NULL ;
2017-05-26 12:39:54 +00:00
return false ; // cancel the rest of the requests
2014-03-15 16:59:03 +00:00
}
DEBFILE ( va ( " Sending file %s (id=%d) to %d \n " , filename , fileid , node ) ) ;
2016-12-31 18:26:33 +00:00
p - > ram = SF_FILE ; // It's a file, we need to close it and free its name once we're done sending it
2014-03-15 16:59:03 +00:00
p - > fileid = fileid ;
2016-12-31 18:26:33 +00:00
p - > next = NULL ; // End of list
filestosend + + ;
2017-05-26 12:39:54 +00:00
return true ;
2014-03-15 16:59:03 +00:00
}
2016-12-31 18:26:33 +00:00
/** Adds a memory block to the file list for a node
*
* \ param node The node to send the memory block to
* \ param data The memory block to send
* \ param size The size of the block in bytes
* \ param freemethod How to free the block after it has been sent
* \ param fileid ? ? ?
* \ sa SV_SendFile
*
*/
void SV_SendRam ( INT32 node , void * data , size_t size , freemethod_t freemethod , UINT8 fileid )
2014-03-15 16:59:03 +00:00
{
2016-12-31 18:26:33 +00:00
filetx_t * * q ; // A pointer to the "next" field of the last file in the list
filetx_t * p ; // The new file request
2014-03-15 16:59:03 +00:00
2016-12-31 18:26:33 +00:00
// Find the last file in the list and set a pointer to its "next" field
2014-03-15 16:59:03 +00:00
q = & transfer [ node ] . txlist ;
while ( * q )
q = & ( ( * q ) - > next ) ;
2016-12-31 18:26:33 +00:00
// Allocate a file request and append it to the file list
2014-03-15 16:59:03 +00:00
p = * q = ( filetx_t * ) malloc ( sizeof ( filetx_t ) ) ;
2016-12-31 18:26:33 +00:00
if ( ! p )
I_Error ( " SV_SendRam: No more memory \n " ) ;
// Initialise with zeros
memset ( p , 0 , sizeof ( filetx_t ) ) ;
p - > ram = freemethod ; // Remember how to free the memory block for when we're done sending it
p - > id . ram = data ;
2014-03-15 16:59:03 +00:00
p - > size = ( UINT32 ) size ;
p - > fileid = fileid ;
2016-12-31 18:26:33 +00:00
p - > next = NULL ; // End of list
2014-03-15 16:59:03 +00:00
2016-12-31 18:26:33 +00:00
DEBFILE ( va ( " Sending ram %p(size:%u) to %d (id=%u) \n " , p - > id . ram , p - > size , node , fileid ) ) ;
2014-03-15 16:59:03 +00:00
2016-12-31 18:26:33 +00:00
filestosend + + ;
2014-03-15 16:59:03 +00:00
}
2016-12-31 18:26:33 +00:00
/** Stops sending a file for a node, and removes the file request from the list,
* either because the file has been fully sent or because the node was disconnected
*
* \ param node The destination
*
*/
static void SV_EndFileSend ( INT32 node )
2014-03-15 16:59:03 +00:00
{
filetx_t * p = transfer [ node ] . txlist ;
2016-12-31 18:26:33 +00:00
// Free the file request according to the freemethod parameter used with SV_SendFile/Ram
2014-03-15 16:59:03 +00:00
switch ( p - > ram )
{
2016-12-31 18:26:33 +00:00
case SF_FILE : // It's a file, close it and free its filename
2017-01-13 19:53:52 +00:00
if ( cv_noticedownload . value )
CONS_Printf ( " Ending file transfer for node %d \n " , node ) ;
2014-03-15 16:59:03 +00:00
if ( transfer [ node ] . currentfile )
fclose ( transfer [ node ] . currentfile ) ;
2016-12-31 18:26:33 +00:00
free ( p - > id . filename ) ;
2014-03-15 16:59:03 +00:00
break ;
2016-12-31 18:26:33 +00:00
case SF_Z_RAM : // It's a memory block allocated with Z_Alloc or the likes, use Z_Free
Z_Free ( p - > id . ram ) ;
2014-03-15 16:59:03 +00:00
break ;
2016-12-31 18:26:33 +00:00
case SF_RAM : // It's a memory block allocated with malloc, use free
free ( p - > id . ram ) ;
case SF_NOFREERAM : // Nothing to free
2014-03-15 16:59:03 +00:00
break ;
}
2016-12-31 18:26:33 +00:00
// Remove the file request from the list
2014-03-15 16:59:03 +00:00
transfer [ node ] . txlist = p - > next ;
free ( p ) ;
2016-12-31 18:26:33 +00:00
// Indicate that the transmission is over
transfer [ node ] . currentfile = NULL ;
filestosend - - ;
2014-03-15 16:59:03 +00:00
}
# define PACKETPERTIC net_bandwidth / (TICRATE*software_MAXPACKETLENGTH)
2016-12-31 18:26:33 +00:00
/** Handles file transmission
*
2017-01-13 19:53:52 +00:00
* \ todo Use an acknowledging method more adapted to file transmission
* The current download speed suffers from lack of ack packets ,
* especially when the one downloading has high latency
2016-12-31 18:26:33 +00:00
*
*/
void SV_FileSendTicker ( void )
2014-03-15 16:59:03 +00:00
{
static INT32 currentnode = 0 ;
filetx_pak * p ;
size_t size ;
filetx_t * f ;
2017-01-13 19:53:52 +00:00
INT32 packetsent , ram , i , j ;
INT32 maxpacketsent ;
2014-03-15 16:59:03 +00:00
2017-01-13 19:53:52 +00:00
if ( ! filestosend ) // No file to send
2014-03-15 16:59:03 +00:00
return ;
2017-01-13 19:53:52 +00:00
if ( cv_downloadspeed . value ) // New (and experimental) behavior
{
packetsent = cv_downloadspeed . value ;
// Don't send more packets than we have free acks
2017-01-13 22:10:00 +00:00
# ifndef NONET
2017-01-13 19:53:52 +00:00
maxpacketsent = Net_GetFreeAcks ( false ) - 5 ; // Let 5 extra acks just in case
2017-01-13 22:10:00 +00:00
# else
maxpacketsent = 1 ;
# endif
2017-01-13 19:53:52 +00:00
if ( packetsent > maxpacketsent & & maxpacketsent > 0 ) // Send at least one packet
packetsent = maxpacketsent ;
}
else // Old behavior
{
packetsent = PACKETPERTIC ;
if ( ! packetsent )
packetsent = 1 ;
}
netbuffer - > packettype = PT_FILEFRAGMENT ;
2014-03-15 16:59:03 +00:00
// (((sendbytes-nowsentbyte)*TICRATE)/(I_GetTime()-starttime)<(UINT32)net_bandwidth)
2016-12-31 18:26:33 +00:00
while ( packetsent - - & & filestosend ! = 0 )
2014-03-15 16:59:03 +00:00
{
2017-01-13 19:53:52 +00:00
for ( i = currentnode , j = 0 ; j < MAXNETNODES ;
i = ( i + 1 ) % MAXNETNODES , j + + )
2014-03-15 16:59:03 +00:00
{
if ( transfer [ i ] . txlist )
goto found ;
}
// no transfer to do
2016-12-31 18:26:33 +00:00
I_Error ( " filestosend=%d but no file to send found \n " , filestosend ) ;
2014-03-15 16:59:03 +00:00
found :
currentnode = ( i + 1 ) % MAXNETNODES ;
f = transfer [ i ] . txlist ;
ram = f - > ram ;
2016-12-31 18:26:33 +00:00
// Open the file if it isn't open yet, or
if ( ! transfer [ i ] . currentfile )
2014-03-15 16:59:03 +00:00
{
2016-12-31 18:26:33 +00:00
if ( ! ram ) // Sending a file
2014-03-15 16:59:03 +00:00
{
long filesize ;
transfer [ i ] . currentfile =
2016-12-31 18:26:33 +00:00
fopen ( f - > id . filename , " rb " ) ;
2014-03-15 16:59:03 +00:00
if ( ! transfer [ i ] . currentfile )
I_Error ( " File %s does not exist " ,
2016-12-31 18:26:33 +00:00
f - > id . filename ) ;
2014-03-15 16:59:03 +00:00
fseek ( transfer [ i ] . currentfile , 0 , SEEK_END ) ;
filesize = ftell ( transfer [ i ] . currentfile ) ;
// Nobody wants to transfer a file bigger
// than 4GB!
if ( filesize > = LONG_MAX )
2016-12-31 18:26:33 +00:00
I_Error ( " filesize of %s is too large " , f - > id . filename ) ;
if ( filesize = = - 1 )
I_Error ( " Error getting filesize of %s " , f - > id . filename ) ;
2014-03-15 16:59:03 +00:00
f - > size = ( UINT32 ) filesize ;
fseek ( transfer [ i ] . currentfile , 0 , SEEK_SET ) ;
}
2016-12-31 18:26:33 +00:00
else // Sending RAM
transfer [ i ] . currentfile = ( FILE * ) 1 ; // Set currentfile to a non-null value to indicate that it is open
2014-03-15 16:59:03 +00:00
transfer [ i ] . position = 0 ;
}
2016-12-31 18:26:33 +00:00
// Build a packet containing a file fragment
2014-03-15 16:59:03 +00:00
p = & netbuffer - > u . filetxpak ;
size = software_MAXPACKETLENGTH - ( FILETXHEADER + BASEPACKETSIZE ) ;
if ( f - > size - transfer [ i ] . position < size )
size = f - > size - transfer [ i ] . position ;
if ( ram )
2016-12-31 18:26:33 +00:00
M_Memcpy ( p - > data , & f - > id . ram [ transfer [ i ] . position ] , size ) ;
2014-03-15 16:59:03 +00:00
else if ( fread ( p - > data , 1 , size , transfer [ i ] . currentfile ) ! = size )
2016-12-31 18:26:33 +00:00
I_Error ( " SV_FileSendTicker: can't read %s byte on %s at %d because %s " , sizeu1 ( size ) , f - > id . filename , transfer [ i ] . position , strerror ( ferror ( transfer [ i ] . currentfile ) ) ) ;
2014-03-15 16:59:03 +00:00
p - > position = LONG ( transfer [ i ] . position ) ;
2016-12-31 18:26:33 +00:00
// Put flag so receiver knows the total size
2014-03-15 16:59:03 +00:00
if ( transfer [ i ] . position + size = = f - > size )
p - > position | = LONG ( 0x80000000 ) ;
p - > fileid = f - > fileid ;
p - > size = SHORT ( ( UINT16 ) size ) ;
2016-12-31 18:26:33 +00:00
// Send the packet
if ( HSendPacket ( i , true , 0 , FILETXHEADER + size ) ) // Reliable SEND
{ // Success
transfer [ i ] . position = ( UINT32 ) ( transfer [ i ] . position + size ) ;
if ( transfer [ i ] . position = = f - > size ) // Finish?
SV_EndFileSend ( i ) ;
}
else
{ // Not sent for some odd reason, retry at next call
2014-03-15 16:59:03 +00:00
if ( ! ram )
2016-12-31 18:26:33 +00:00
fseek ( transfer [ i ] . currentfile , transfer [ i ] . position , SEEK_SET ) ;
// Exit the while (can't send this one so why should i send the next?)
2014-03-15 16:59:03 +00:00
break ;
}
}
}
void Got_Filetxpak ( void )
{
INT32 filenum = netbuffer - > u . filetxpak . fileid ;
2017-01-13 19:53:52 +00:00
fileneeded_t * file = & fileneeded [ filenum ] ;
char * filename = file - > filename ;
2014-03-15 16:59:03 +00:00
static INT32 filetime = 0 ;
2017-01-13 19:53:52 +00:00
if ( ! ( strcmp ( filename , " srb2.srb " )
& & strcmp ( filename , " srb2.wad " )
& & strcmp ( filename , " patch.dta " )
2018-11-16 00:18:04 +00:00
//&& strcmp(filename, "music.dta")
& & strcmp ( filename , " gfx.kart " )
& & strcmp ( filename , " textures.kart " )
& & strcmp ( filename , " chars.kart " )
& & strcmp ( filename , " maps.kart " )
& & strcmp ( filename , " sounds.kart " )
& & strcmp ( filename , " music.kart " )
& & strcmp ( filename , " patch.kart " )
2017-01-13 19:53:52 +00:00
) )
I_Error ( " Tried to download \" %s \" " , filename ) ;
2014-03-15 16:59:03 +00:00
if ( filenum > = fileneedednum )
{
2016-12-31 18:26:33 +00:00
DEBFILE ( va ( " fileframent not needed %d>%d \n " , filenum , fileneedednum ) ) ;
2017-01-13 19:53:52 +00:00
//I_Error("Received an unneeded file fragment (file id received: %d, file id needed: %d)\n", filenum, fileneedednum);
2014-03-15 16:59:03 +00:00
return ;
}
2017-01-13 19:53:52 +00:00
if ( file - > status = = FS_REQUESTED )
2014-03-15 16:59:03 +00:00
{
2017-01-13 19:53:52 +00:00
if ( file - > file )
2016-12-31 18:26:33 +00:00
I_Error ( " Got_Filetxpak: already open file \n " ) ;
2017-01-13 19:53:52 +00:00
file - > file = fopen ( filename , " wb " ) ;
if ( ! file - > file )
I_Error ( " Can't create file %s: %s " , filename , strerror ( errno ) ) ;
CONS_Printf ( " \r %s... \n " , filename ) ;
file - > currentsize = 0 ;
file - > status = FS_DOWNLOADING ;
2014-03-15 16:59:03 +00:00
}
2017-01-13 19:53:52 +00:00
if ( file - > status = = FS_DOWNLOADING )
2014-03-15 16:59:03 +00:00
{
UINT32 pos = LONG ( netbuffer - > u . filetxpak . position ) ;
UINT16 size = SHORT ( netbuffer - > u . filetxpak . size ) ;
2016-12-31 18:26:33 +00:00
// Use a special trick to know when the file is complete (not always used)
// WARNING: file fragments can arrive out of order so don't stop yet!
2014-03-15 16:59:03 +00:00
if ( pos & 0x80000000 )
{
pos & = ~ 0x80000000 ;
2017-01-13 19:53:52 +00:00
file - > totalsize = pos + size ;
2014-03-15 16:59:03 +00:00
}
2016-12-31 18:26:33 +00:00
// We can receive packet in the wrong order, anyway all os support gaped file
2017-01-13 19:53:52 +00:00
fseek ( file - > file , pos , SEEK_SET ) ;
if ( fwrite ( netbuffer - > u . filetxpak . data , size , 1 , file - > file ) ! = 1 )
I_Error ( " Can't write to %s: %s \n " , filename , strerror ( ferror ( file - > file ) ) ) ;
file - > currentsize + = size ;
2014-03-15 16:59:03 +00:00
2016-12-31 18:26:33 +00:00
// Finished?
2017-01-13 19:53:52 +00:00
if ( file - > currentsize = = file - > totalsize )
2014-03-15 16:59:03 +00:00
{
2017-01-13 19:53:52 +00:00
fclose ( file - > file ) ;
file - > file = NULL ;
file - > status = FS_FOUND ;
2014-03-15 16:59:03 +00:00
CONS_Printf ( M_GetText ( " Downloading %s...(done) \n " ) ,
2017-01-13 19:53:52 +00:00
filename ) ;
2014-03-15 16:59:03 +00:00
}
}
else
2017-01-13 19:53:52 +00:00
{
const char * s ;
switch ( file - > status )
{
case FS_NOTFOUND :
s = " FS_NOTFOUND " ;
break ;
case FS_FOUND :
s = " FS_FOUND " ;
break ;
case FS_OPEN :
s = " FS_OPEN " ;
break ;
case FS_MD5SUMBAD :
s = " FS_MD5SUMBAD " ;
break ;
default :
s = " unknown " ;
break ;
}
I_Error ( " Received a file not requested (file id: %d, file status: %s) \n " , filenum , s ) ;
}
2016-12-31 18:26:33 +00:00
// Send ack back quickly
2014-03-15 16:59:03 +00:00
if ( + + filetime = = 3 )
{
Net_SendAcks ( servernode ) ;
filetime = 0 ;
}
# ifdef CLIENT_LOADINGSCREEN
lastfilenum = filenum ;
# endif
}
2017-01-13 19:53:52 +00:00
/** \brief Checks if a node is downloading a file
2016-12-31 18:26:33 +00:00
*
2017-01-13 19:53:52 +00:00
* \ param node The node to check for
* \ return True if the node is downloading a file
2016-12-31 18:26:33 +00:00
*
*/
2017-01-13 19:53:52 +00:00
boolean SV_SendingFile ( INT32 node )
{
return transfer [ node ] . txlist ! = NULL ;
}
/** Cancels all file requests for a node
*
* \ param node The destination
* \ sa SV_EndFileSend
*
*/
2016-12-31 18:26:33 +00:00
void SV_AbortSendFiles ( INT32 node )
2014-03-15 16:59:03 +00:00
{
while ( transfer [ node ] . txlist )
2016-12-31 18:26:33 +00:00
SV_EndFileSend ( node ) ;
2014-03-15 16:59:03 +00:00
}
void CloseNetFile ( void )
{
INT32 i ;
2016-12-31 18:26:33 +00:00
// Is sending?
2014-03-15 16:59:03 +00:00
for ( i = 0 ; i < MAXNETNODES ; i + + )
2016-12-31 18:26:33 +00:00
SV_AbortSendFiles ( i ) ;
2014-03-15 16:59:03 +00:00
2016-12-31 18:26:33 +00:00
// Receiving a file?
2014-03-15 16:59:03 +00:00
for ( i = 0 ; i < MAX_WADFILES ; i + + )
2016-12-31 18:26:33 +00:00
if ( fileneeded [ i ] . status = = FS_DOWNLOADING & & fileneeded [ i ] . file )
2014-03-15 16:59:03 +00:00
{
2016-12-31 18:26:33 +00:00
fclose ( fileneeded [ i ] . file ) ;
// File is not complete delete it
2014-03-15 16:59:03 +00:00
remove ( fileneeded [ i ] . filename ) ;
}
2016-12-31 18:26:33 +00:00
// Remove PT_FILEFRAGMENT from acknowledge list
2014-03-15 16:59:03 +00:00
Net_AbortPacketType ( PT_FILEFRAGMENT ) ;
}
2016-12-31 18:26:33 +00:00
// Functions cut and pasted from Doomatic :)
2014-03-15 16:59:03 +00:00
void nameonly ( char * s )
{
size_t j , len ;
void * ns ;
for ( j = strlen ( s ) ; j ! = ( size_t ) - 1 ; j - - )
if ( ( s [ j ] = = ' \\ ' ) | | ( s [ j ] = = ' : ' ) | | ( s [ j ] = = ' / ' ) )
{
ns = & ( s [ j + 1 ] ) ;
len = strlen ( ns ) ;
2018-08-28 20:08:47 +00:00
#if 0
2014-03-15 16:59:03 +00:00
M_Memcpy ( s , ns , len + 1 ) ;
2018-08-28 20:08:47 +00:00
# else
2014-03-15 16:59:03 +00:00
memmove ( s , ns , len + 1 ) ;
2018-08-28 20:08:47 +00:00
# endif
2014-03-15 16:59:03 +00:00
return ;
}
}
// Returns the length in characters of the last element of a path.
size_t nameonlylength ( const char * s )
{
size_t j , len = strlen ( s ) ;
for ( j = len ; j ! = ( size_t ) - 1 ; j - - )
if ( ( s [ j ] = = ' \\ ' ) | | ( s [ j ] = = ' : ' ) | | ( s [ j ] = = ' / ' ) )
return len - j - 1 ;
return len ;
}
# ifndef O_BINARY
# define O_BINARY 0
# endif
filestatus_t checkfilemd5 ( char * filename , const UINT8 * wantedmd5sum )
{
# if defined (NOMD5) || defined (_arch_dreamcast)
( void ) wantedmd5sum ;
( void ) filename ;
# else
FILE * fhandle ;
UINT8 md5sum [ 16 ] ;
if ( ! wantedmd5sum )
return FS_FOUND ;
fhandle = fopen ( filename , " rb " ) ;
if ( fhandle )
{
md5_stream ( fhandle , md5sum ) ;
fclose ( fhandle ) ;
if ( ! memcmp ( wantedmd5sum , md5sum , 16 ) )
return FS_FOUND ;
return FS_MD5SUMBAD ;
}
I_Error ( " Couldn't open %s for md5 check " , filename ) ;
# endif
return FS_FOUND ; // will never happen, but makes the compiler shut up
}
2018-04-15 21:00:31 +00:00
// Rewritten by Monster Iestyn to be less stupid
// Note: if completepath is true, "filename" is modified, but only if FS_FOUND is going to be returned
// (Don't worry about WinCE's version of filesearch, nobody cares about that OS anymore)
2014-03-15 16:59:03 +00:00
filestatus_t findfile ( char * filename , const UINT8 * wantedmd5sum , boolean completepath )
{
2018-04-15 21:00:31 +00:00
filestatus_t homecheck ; // store result of last file search
boolean badmd5 = false ; // store whether md5 was bad from either of the first two searches (if nothing was found in the third)
2014-03-15 16:59:03 +00:00
2018-04-15 21:00:31 +00:00
// first, check SRB2's "home" directory
homecheck = filesearch ( filename , srb2home , wantedmd5sum , completepath , 10 ) ;
2014-03-15 16:59:03 +00:00
2018-04-15 21:00:31 +00:00
if ( homecheck = = FS_FOUND ) // we found the file, so return that we have :)
return FS_FOUND ;
else if ( homecheck = = FS_MD5SUMBAD ) / / file has a bad md5 ; move on and look for a file with the right md5
badmd5 = true ;
// if not found at all, just move on without doing anything
// next, check SRB2's "path" directory
homecheck = filesearch ( filename , srb2path , wantedmd5sum , completepath , 10 ) ;
if ( homecheck = = FS_FOUND ) // we found the file, so return that we have :)
return FS_FOUND ;
else if ( homecheck = = FS_MD5SUMBAD ) / / file has a bad md5 ; move on and look for a file with the right md5
badmd5 = true ;
// if not found at all, just move on without doing anything
// finally check "." directory
2014-03-15 16:59:03 +00:00
# ifdef _arch_dreamcast
2018-04-15 21:00:31 +00:00
homecheck = filesearch ( filename , " /cd " , wantedmd5sum , completepath , 10 ) ;
2014-03-15 16:59:03 +00:00
# else
2018-04-15 21:00:31 +00:00
homecheck = filesearch ( filename , " . " , wantedmd5sum , completepath , 10 ) ;
2014-03-15 16:59:03 +00:00
# endif
2018-04-15 21:00:31 +00:00
if ( homecheck ! = FS_NOTFOUND ) // if not found this time, fall back on the below return statement
return homecheck ; // otherwise return the result we got
return ( badmd5 ? FS_MD5SUMBAD : FS_NOTFOUND ) ; // md5 sum bad or file not found
2014-03-15 16:59:03 +00:00
}