2008-07-22 23:28:28 +00:00
|
|
|
/*
|
|
|
|
* This file Copyright (C) 2008 Charles Kerr <charles@rebelbase.com>
|
2006-07-16 19:39:23 +00:00
|
|
|
*
|
2008-07-22 23:28:28 +00:00
|
|
|
* This file is licensed by the GPL version 2. Works owned by the
|
|
|
|
* Transmission project are granted a special exemption to clause 2(b)
|
2008-09-23 19:11:04 +00:00
|
|
|
* so that the bulk of its code can remain under the MIT license.
|
2008-07-22 23:28:28 +00:00
|
|
|
* This exemption does not extend to derived works not owned by
|
|
|
|
* the Transmission project.
|
2006-07-16 19:39:23 +00:00
|
|
|
*
|
2008-07-22 23:28:28 +00:00
|
|
|
* $Id$
|
|
|
|
*/
|
2006-07-16 19:39:23 +00:00
|
|
|
|
2007-08-04 01:17:39 +00:00
|
|
|
#ifdef __BEOS__
|
2008-09-23 19:11:04 +00:00
|
|
|
#include <signal.h>
|
|
|
|
#include <fs_info.h>
|
|
|
|
#include <FindDirectory.h>
|
|
|
|
#include <kernel/OS.h>
|
|
|
|
#define BEOS_MAX_THREADS 256
|
|
|
|
#elif defined( WIN32 )
|
|
|
|
#include <windows.h>
|
2008-10-13 22:26:02 +00:00
|
|
|
#include <shlobj.h> /* for CSIDL_APPDATA, CSIDL_MYDOCUMENTS */
|
2007-07-30 15:27:52 +00:00
|
|
|
#else
|
2008-09-23 19:11:04 +00:00
|
|
|
#ifdef SYS_DARWIN
|
|
|
|
#include <CoreFoundation/CoreFoundation.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define _XOPEN_SOURCE 500 /* needed for recursive locks. */
|
|
|
|
#ifndef __USE_UNIX98
|
2007-12-02 17:15:52 +00:00
|
|
|
#define __USE_UNIX98 /* some older Linuxes need it spelt out for them */
|
2008-09-23 19:11:04 +00:00
|
|
|
#endif
|
|
|
|
#include <pthread.h>
|
2006-07-16 19:39:23 +00:00
|
|
|
#endif
|
2007-07-30 15:27:52 +00:00
|
|
|
|
2007-12-07 19:25:54 +00:00
|
|
|
#include <assert.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2008-07-11 04:11:22 +00:00
|
|
|
#include <sys/stat.h>
|
2006-07-16 19:39:23 +00:00
|
|
|
#include <sys/types.h>
|
2008-10-13 22:26:02 +00:00
|
|
|
#ifdef WIN32
|
|
|
|
#include <libgen.h>
|
|
|
|
#endif
|
2006-07-16 19:39:23 +00:00
|
|
|
#include <dirent.h>
|
2007-11-27 15:39:59 +00:00
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h> /* getuid getpid close */
|
2007-07-29 18:11:21 +00:00
|
|
|
|
2006-07-16 19:39:23 +00:00
|
|
|
#include "transmission.h"
|
2008-07-11 04:07:14 +00:00
|
|
|
#include "list.h"
|
2007-07-31 14:26:44 +00:00
|
|
|
#include "platform.h"
|
2007-07-30 15:27:52 +00:00
|
|
|
#include "utils.h"
|
|
|
|
|
|
|
|
/***
|
|
|
|
**** THREADS
|
|
|
|
***/
|
|
|
|
|
2007-10-01 15:17:15 +00:00
|
|
|
#ifdef __BEOS__
|
|
|
|
typedef thread_id tr_thread_id;
|
2008-09-23 19:11:04 +00:00
|
|
|
#elif defined( WIN32 )
|
2007-10-01 15:17:15 +00:00
|
|
|
typedef DWORD tr_thread_id;
|
|
|
|
#else
|
|
|
|
typedef pthread_t tr_thread_id;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static tr_thread_id
|
|
|
|
tr_getCurrentThread( void )
|
|
|
|
{
|
|
|
|
#ifdef __BEOS__
|
|
|
|
return find_thread( NULL );
|
2008-09-23 19:11:04 +00:00
|
|
|
#elif defined( WIN32 )
|
|
|
|
return GetCurrentThreadId( );
|
2007-10-01 15:17:15 +00:00
|
|
|
#else
|
|
|
|
return pthread_self( );
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2008-09-23 19:11:04 +00:00
|
|
|
tr_areThreadsEqual( tr_thread_id a,
|
|
|
|
tr_thread_id b )
|
2007-10-01 15:17:15 +00:00
|
|
|
{
|
|
|
|
#ifdef __BEOS__
|
|
|
|
return a == b;
|
2008-09-23 19:11:04 +00:00
|
|
|
#elif defined( WIN32 )
|
2007-10-01 15:17:15 +00:00
|
|
|
return a == b;
|
|
|
|
#else
|
|
|
|
return pthread_equal( a, b );
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2007-09-20 16:32:01 +00:00
|
|
|
struct tr_thread
|
2007-07-30 15:27:52 +00:00
|
|
|
{
|
2008-09-23 19:11:04 +00:00
|
|
|
void ( * func )( void * );
|
|
|
|
void * arg;
|
|
|
|
tr_thread_id thread;
|
2008-02-28 16:40:31 +00:00
|
|
|
#ifdef WIN32
|
2008-09-23 19:11:04 +00:00
|
|
|
HANDLE thread_handle;
|
2007-07-30 15:27:52 +00:00
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2007-10-01 15:17:15 +00:00
|
|
|
int
|
2008-09-23 19:11:04 +00:00
|
|
|
tr_amInThread( const tr_thread * t )
|
2007-10-01 15:17:15 +00:00
|
|
|
{
|
2008-09-23 19:11:04 +00:00
|
|
|
return tr_areThreadsEqual( tr_getCurrentThread( ), t->thread );
|
2007-10-01 15:17:15 +00:00
|
|
|
}
|
|
|
|
|
2007-07-31 19:56:40 +00:00
|
|
|
#ifdef WIN32
|
2008-09-23 19:11:04 +00:00
|
|
|
#define ThreadFuncReturnType unsigned WINAPI
|
2007-07-31 19:56:40 +00:00
|
|
|
#else
|
2008-09-23 19:11:04 +00:00
|
|
|
#define ThreadFuncReturnType void
|
2007-07-31 19:56:40 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
static ThreadFuncReturnType
|
2007-07-30 15:27:52 +00:00
|
|
|
ThreadFunc( void * _t )
|
|
|
|
{
|
2007-09-20 16:32:01 +00:00
|
|
|
tr_thread * t = _t;
|
2007-07-30 15:27:52 +00:00
|
|
|
|
2007-08-04 01:17:39 +00:00
|
|
|
#ifdef __BEOS__
|
2007-07-30 15:27:52 +00:00
|
|
|
/* This is required because on BeOS, SIGINT is sent to each thread,
|
|
|
|
which kills them not nicely */
|
|
|
|
signal( SIGINT, SIG_IGN );
|
|
|
|
#endif
|
|
|
|
|
|
|
|
t->func( t->arg );
|
2007-07-31 19:56:40 +00:00
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
_endthreadex( 0 );
|
|
|
|
return 0;
|
|
|
|
#endif
|
2007-07-30 15:27:52 +00:00
|
|
|
}
|
|
|
|
|
2007-09-20 16:32:01 +00:00
|
|
|
tr_thread *
|
2008-09-23 19:11:04 +00:00
|
|
|
tr_threadNew( void ( *func )(void *),
|
|
|
|
void * arg )
|
2007-07-30 15:27:52 +00:00
|
|
|
{
|
2007-09-20 16:32:01 +00:00
|
|
|
tr_thread * t = tr_new0( tr_thread, 1 );
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2008-02-29 01:58:46 +00:00
|
|
|
t->func = func;
|
|
|
|
t->arg = arg;
|
2007-07-30 15:27:52 +00:00
|
|
|
|
2007-08-04 01:17:39 +00:00
|
|
|
#ifdef __BEOS__
|
2008-09-23 19:11:04 +00:00
|
|
|
t->thread =
|
|
|
|
spawn_thread( (void*)ThreadFunc, "beos thread", B_NORMAL_PRIORITY,
|
|
|
|
t );
|
2007-07-30 15:27:52 +00:00
|
|
|
resume_thread( t->thread );
|
2008-09-23 19:11:04 +00:00
|
|
|
#elif defined( WIN32 )
|
2008-02-29 01:58:46 +00:00
|
|
|
{
|
|
|
|
unsigned int id;
|
2008-09-23 19:11:04 +00:00
|
|
|
t->thread_handle =
|
|
|
|
(HANDLE) _beginthreadex( NULL, 0, &ThreadFunc, t, 0,
|
|
|
|
&id );
|
2008-02-29 01:58:46 +00:00
|
|
|
t->thread = (DWORD) id;
|
|
|
|
}
|
2007-07-30 15:27:52 +00:00
|
|
|
#else
|
2008-09-23 19:11:04 +00:00
|
|
|
pthread_create( &t->thread, NULL, ( void * ( * )(
|
|
|
|
void * ) )ThreadFunc, t );
|
2007-07-30 15:27:52 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
return t;
|
|
|
|
}
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2007-07-30 15:27:52 +00:00
|
|
|
/***
|
|
|
|
**** LOCKS
|
|
|
|
***/
|
|
|
|
|
2007-09-20 16:32:01 +00:00
|
|
|
struct tr_lock
|
2007-07-30 15:27:52 +00:00
|
|
|
{
|
2008-09-23 19:11:04 +00:00
|
|
|
int depth;
|
2007-08-04 01:17:39 +00:00
|
|
|
#ifdef __BEOS__
|
2008-09-23 19:11:04 +00:00
|
|
|
sem_id lock;
|
|
|
|
thread_id lockThread;
|
|
|
|
#elif defined( WIN32 )
|
|
|
|
CRITICAL_SECTION lock;
|
|
|
|
DWORD lockThread;
|
2007-07-30 15:27:52 +00:00
|
|
|
#else
|
2008-09-23 19:11:04 +00:00
|
|
|
pthread_mutex_t lock;
|
|
|
|
pthread_t lockThread;
|
2007-07-30 15:27:52 +00:00
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2007-09-20 16:32:01 +00:00
|
|
|
tr_lock*
|
2007-07-30 15:27:52 +00:00
|
|
|
tr_lockNew( void )
|
|
|
|
{
|
2008-09-23 19:11:04 +00:00
|
|
|
tr_lock * l = tr_new0( tr_lock, 1 );
|
2007-07-30 15:27:52 +00:00
|
|
|
|
2007-08-04 01:17:39 +00:00
|
|
|
#ifdef __BEOS__
|
2007-07-30 15:27:52 +00:00
|
|
|
l->lock = create_sem( 1, "" );
|
2008-09-23 19:11:04 +00:00
|
|
|
#elif defined( WIN32 )
|
2008-04-18 23:17:40 +00:00
|
|
|
InitializeCriticalSection( &l->lock ); /* supports recursion */
|
2007-07-30 15:27:52 +00:00
|
|
|
#else
|
2007-12-02 16:35:44 +00:00
|
|
|
pthread_mutexattr_t attr;
|
|
|
|
pthread_mutexattr_init( &attr );
|
|
|
|
pthread_mutexattr_settype( &attr, PTHREAD_MUTEX_RECURSIVE );
|
|
|
|
pthread_mutex_init( &l->lock, &attr );
|
2007-07-30 15:27:52 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
return l;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2007-09-20 16:32:01 +00:00
|
|
|
tr_lockFree( tr_lock * l )
|
2007-07-30 15:27:52 +00:00
|
|
|
{
|
2007-08-04 01:17:39 +00:00
|
|
|
#ifdef __BEOS__
|
2007-07-30 15:27:52 +00:00
|
|
|
delete_sem( l->lock );
|
2008-09-23 19:11:04 +00:00
|
|
|
#elif defined( WIN32 )
|
2007-07-31 19:56:40 +00:00
|
|
|
DeleteCriticalSection( &l->lock );
|
2007-07-30 15:27:52 +00:00
|
|
|
#else
|
|
|
|
pthread_mutex_destroy( &l->lock );
|
|
|
|
#endif
|
|
|
|
tr_free( l );
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2007-09-20 16:32:01 +00:00
|
|
|
tr_lockLock( tr_lock * l )
|
2007-07-30 15:27:52 +00:00
|
|
|
{
|
2007-08-04 01:17:39 +00:00
|
|
|
#ifdef __BEOS__
|
2007-12-02 16:35:44 +00:00
|
|
|
acquire_sem( l->lock );
|
2008-09-23 19:11:04 +00:00
|
|
|
#elif defined( WIN32 )
|
2007-12-02 16:35:44 +00:00
|
|
|
EnterCriticalSection( &l->lock );
|
2007-07-30 15:27:52 +00:00
|
|
|
#else
|
2007-12-02 16:35:44 +00:00
|
|
|
pthread_mutex_lock( &l->lock );
|
2007-07-30 15:27:52 +00:00
|
|
|
#endif
|
2007-12-03 04:06:45 +00:00
|
|
|
assert( l->depth >= 0 );
|
|
|
|
if( l->depth )
|
2008-09-23 19:11:04 +00:00
|
|
|
assert( tr_areThreadsEqual( l->lockThread, tr_getCurrentThread( ) ) );
|
2007-12-02 18:20:37 +00:00
|
|
|
l->lockThread = tr_getCurrentThread( );
|
2007-12-02 16:35:44 +00:00
|
|
|
++l->depth;
|
2007-10-01 15:17:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
tr_lockHave( const tr_lock * l )
|
|
|
|
{
|
|
|
|
return ( l->depth > 0 )
|
2008-09-23 19:11:04 +00:00
|
|
|
&& ( tr_areThreadsEqual( l->lockThread, tr_getCurrentThread( ) ) );
|
2007-07-30 15:27:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2007-09-20 16:32:01 +00:00
|
|
|
tr_lockUnlock( tr_lock * l )
|
2007-07-30 15:27:52 +00:00
|
|
|
{
|
2007-12-02 17:15:52 +00:00
|
|
|
assert( l->depth > 0 );
|
2008-09-23 19:11:04 +00:00
|
|
|
assert( tr_areThreadsEqual( l->lockThread, tr_getCurrentThread( ) ) );
|
2007-10-01 15:17:15 +00:00
|
|
|
|
2007-12-02 18:20:37 +00:00
|
|
|
--l->depth;
|
|
|
|
assert( l->depth >= 0 );
|
2007-08-04 01:17:39 +00:00
|
|
|
#ifdef __BEOS__
|
2007-12-02 16:35:44 +00:00
|
|
|
release_sem( l->lock );
|
2008-09-23 19:11:04 +00:00
|
|
|
#elif defined( WIN32 )
|
2007-12-02 16:35:44 +00:00
|
|
|
LeaveCriticalSection( &l->lock );
|
2007-07-30 15:27:52 +00:00
|
|
|
#else
|
2007-12-02 16:35:44 +00:00
|
|
|
pthread_mutex_unlock( &l->lock );
|
2007-07-30 15:27:52 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/***
|
|
|
|
**** PATHS
|
|
|
|
***/
|
2006-07-16 19:39:23 +00:00
|
|
|
|
2008-09-23 19:11:04 +00:00
|
|
|
#if !defined( WIN32 ) && !defined( __BEOS__ ) && !defined( __AMIGAOS4__ )
|
|
|
|
#include <pwd.h>
|
2007-08-02 19:43:29 +00:00
|
|
|
#endif
|
2006-10-13 06:29:26 +00:00
|
|
|
|
2007-12-15 22:22:30 +00:00
|
|
|
static const char *
|
2008-04-05 20:12:11 +00:00
|
|
|
getHomeDir( void )
|
2006-10-13 06:29:26 +00:00
|
|
|
{
|
2008-04-05 20:12:11 +00:00
|
|
|
static char * home = NULL;
|
2006-10-13 06:29:26 +00:00
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
if( !home )
|
|
|
|
{
|
|
|
|
home = tr_strdup( getenv( "HOME" ) );
|
2006-10-13 06:29:26 +00:00
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
if( !home )
|
|
|
|
{
|
2007-08-02 19:43:29 +00:00
|
|
|
#ifdef WIN32
|
2008-10-13 22:26:02 +00:00
|
|
|
SHGetFolderPath( NULL, CSIDL_MYDOCUMENTS, NULL, 0, home );
|
2008-09-23 19:11:04 +00:00
|
|
|
#elif defined( __BEOS__ ) || defined( __AMIGAOS4__ )
|
2008-04-05 20:12:11 +00:00
|
|
|
home = tr_strdup( "" );
|
2007-08-02 19:43:29 +00:00
|
|
|
#else
|
2008-09-23 19:11:04 +00:00
|
|
|
struct passwd * pw = getpwuid( getuid( ) );
|
2008-04-05 20:12:11 +00:00
|
|
|
if( pw )
|
|
|
|
home = tr_strdup( pw->pw_dir );
|
2008-04-18 23:17:40 +00:00
|
|
|
endpwent( );
|
2007-08-02 19:43:29 +00:00
|
|
|
#endif
|
2006-07-16 19:39:23 +00:00
|
|
|
}
|
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
if( !home )
|
|
|
|
home = tr_strdup( "" );
|
2006-07-16 19:39:23 +00:00
|
|
|
}
|
2008-04-05 20:12:11 +00:00
|
|
|
|
|
|
|
return home;
|
2006-07-16 19:39:23 +00:00
|
|
|
}
|
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
static const char *
|
|
|
|
getOldConfigDir( void )
|
2006-07-16 19:39:23 +00:00
|
|
|
{
|
2008-04-05 20:12:11 +00:00
|
|
|
static char * path = NULL;
|
2006-07-16 19:39:23 +00:00
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
if( !path )
|
2007-12-15 22:22:30 +00:00
|
|
|
{
|
2007-08-04 01:17:39 +00:00
|
|
|
#ifdef __BEOS__
|
2008-10-14 03:03:29 +00:00
|
|
|
char buf[MAX_PATH_LENGTH];
|
2007-12-15 22:22:30 +00:00
|
|
|
find_directory( B_USER_SETTINGS_DIRECTORY,
|
2008-09-23 19:11:04 +00:00
|
|
|
dev_for_path( "/boot" ), true,
|
|
|
|
buf, sizeof( buf ) );
|
2008-10-14 03:03:29 +00:00
|
|
|
path = tr_buildPath( buf, "Transmission", NULL );
|
2006-07-16 19:39:23 +00:00
|
|
|
#elif defined( SYS_DARWIN )
|
2008-10-14 03:03:29 +00:00
|
|
|
path = tr_buildPath( getHomeDir( ), "Library",
|
|
|
|
"Application Support",
|
|
|
|
"Transmission", NULL );
|
2008-09-23 19:11:04 +00:00
|
|
|
#elif defined( __AMIGAOS4__ )
|
2008-10-14 03:03:29 +00:00
|
|
|
path = tr_strdup( "PROGDIR:.transmission" );
|
2008-09-23 19:11:04 +00:00
|
|
|
#elif defined( WIN32 )
|
2007-12-15 22:22:30 +00:00
|
|
|
char appdata[MAX_PATH_LENGTH];
|
|
|
|
SHGetFolderPath( NULL, CSIDL_APPDATA, NULL, 0, appdata );
|
2008-10-14 03:03:29 +00:00
|
|
|
path = tr_buildPath( appdata, "Transmission", NULL );
|
2006-07-16 19:39:23 +00:00
|
|
|
#else
|
2008-10-14 03:03:29 +00:00
|
|
|
path = tr_buildPath( getHomeDir( ), ".transmission", NULL );
|
2006-07-16 19:39:23 +00:00
|
|
|
#endif
|
2007-12-15 22:22:30 +00:00
|
|
|
}
|
2006-07-16 19:39:23 +00:00
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
return path;
|
|
|
|
}
|
2006-07-16 19:39:23 +00:00
|
|
|
|
2008-10-14 03:03:29 +00:00
|
|
|
#if defined(SYS_DARWIN) || defined(WIN32)
|
|
|
|
#define RESUME_SUBDIR "Resume"
|
|
|
|
#define TORRENT_SUBDIR "Torrents"
|
|
|
|
#else
|
|
|
|
#define RESUME_SUBDIR "resume"
|
|
|
|
#define TORRENT_SUBDIR "torrents"
|
|
|
|
#endif
|
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
static const char *
|
|
|
|
getOldTorrentsDir( void )
|
|
|
|
{
|
|
|
|
static char * path = NULL;
|
|
|
|
|
|
|
|
if( !path )
|
2008-10-14 03:03:29 +00:00
|
|
|
path = tr_buildPath( getOldConfigDir( ), TORRENT_SUBDIR, NULL );
|
2006-07-16 19:39:23 +00:00
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
return path;
|
|
|
|
}
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
static const char *
|
|
|
|
getOldCacheDir( void )
|
2006-07-16 19:39:23 +00:00
|
|
|
{
|
2008-04-05 20:12:11 +00:00
|
|
|
static char * path = NULL;
|
2006-07-16 19:39:23 +00:00
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
if( !path )
|
|
|
|
{
|
2008-09-23 19:11:04 +00:00
|
|
|
#if defined( __BEOS__ ) || defined( WIN32 )
|
2008-10-14 03:03:29 +00:00
|
|
|
path = tr_buildPath( getOldConfigDir( ), "Cache", NULL );
|
2006-07-16 19:39:23 +00:00
|
|
|
#elif defined( SYS_DARWIN )
|
2008-10-14 03:03:29 +00:00
|
|
|
path = tr_buildPath( getHomeDir( ), "Library", "Caches", "Transmission", NULL );
|
2006-07-16 19:39:23 +00:00
|
|
|
#else
|
2008-10-14 03:03:29 +00:00
|
|
|
path = tr_buildPath( getOldConfigDir( ), "cache", NULL );
|
2006-07-16 19:39:23 +00:00
|
|
|
#endif
|
2008-04-05 20:12:11 +00:00
|
|
|
}
|
2006-07-16 19:39:23 +00:00
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2008-09-23 19:11:04 +00:00
|
|
|
moveFiles( const char * oldDir,
|
|
|
|
const char * newDir )
|
2008-04-05 20:12:11 +00:00
|
|
|
{
|
|
|
|
if( oldDir && newDir && strcmp( oldDir, newDir ) )
|
|
|
|
{
|
|
|
|
DIR * dirh = opendir( oldDir );
|
|
|
|
if( dirh )
|
|
|
|
{
|
2008-09-23 19:11:04 +00:00
|
|
|
int count = 0;
|
2008-04-05 20:12:11 +00:00
|
|
|
struct dirent * dirp;
|
2008-09-23 19:11:04 +00:00
|
|
|
while( ( dirp = readdir( dirh ) ) )
|
2008-04-05 20:12:11 +00:00
|
|
|
{
|
2008-09-23 19:11:04 +00:00
|
|
|
if( strcmp( dirp->d_name,
|
|
|
|
"." ) && strcmp( dirp->d_name, ".." ) )
|
2008-04-05 20:12:11 +00:00
|
|
|
{
|
2008-10-14 03:03:29 +00:00
|
|
|
char * o = tr_buildPath( oldDir, dirp->d_name, NULL );
|
|
|
|
char * n = tr_buildPath( newDir, dirp->d_name, NULL );
|
2008-04-05 20:12:11 +00:00
|
|
|
rename( o, n );
|
|
|
|
++count;
|
2008-10-14 03:03:29 +00:00
|
|
|
tr_free( n );
|
|
|
|
tr_free( o );
|
2008-04-05 20:12:11 +00:00
|
|
|
}
|
|
|
|
}
|
2008-04-10 02:53:03 +00:00
|
|
|
|
|
|
|
if( count )
|
|
|
|
tr_inf( _( "Migrated %1$d files from \"%2$s\" to \"%3$s\"" ),
|
|
|
|
count, oldDir, newDir );
|
2008-04-05 20:12:11 +00:00
|
|
|
closedir( dirh );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
migrateFiles( const tr_handle * handle )
|
|
|
|
{
|
|
|
|
static int migrated = FALSE;
|
|
|
|
|
|
|
|
if( !migrated )
|
|
|
|
{
|
|
|
|
const char * oldDir;
|
|
|
|
const char * newDir;
|
|
|
|
migrated = TRUE;
|
|
|
|
|
|
|
|
oldDir = getOldTorrentsDir( );
|
|
|
|
newDir = tr_getTorrentDir( handle );
|
|
|
|
moveFiles( oldDir, newDir );
|
|
|
|
|
|
|
|
oldDir = getOldCacheDir( );
|
|
|
|
newDir = tr_getResumeDir( handle );
|
|
|
|
moveFiles( oldDir, newDir );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2008-09-23 19:11:04 +00:00
|
|
|
tr_setConfigDir( tr_handle * handle,
|
|
|
|
const char * configDir )
|
2008-04-05 20:12:11 +00:00
|
|
|
{
|
2008-10-14 03:03:29 +00:00
|
|
|
char * path;
|
2008-04-05 20:12:11 +00:00
|
|
|
|
|
|
|
handle->configDir = tr_strdup( configDir );
|
|
|
|
|
2008-10-14 03:03:29 +00:00
|
|
|
path = tr_buildPath( configDir, RESUME_SUBDIR, NULL );
|
|
|
|
tr_mkdirp( path, 0777 );
|
|
|
|
handle->resumeDir = path;
|
2008-04-05 20:12:11 +00:00
|
|
|
|
2008-10-14 03:03:29 +00:00
|
|
|
path = tr_buildPath( configDir, TORRENT_SUBDIR, NULL );
|
|
|
|
tr_mkdirp( path, 0777 );
|
|
|
|
handle->torrentDir = path;
|
2006-07-16 19:39:23 +00:00
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
migrateFiles( handle );
|
|
|
|
}
|
2006-07-16 19:39:23 +00:00
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
const char *
|
2008-05-18 16:44:30 +00:00
|
|
|
tr_sessionGetConfigDir( const tr_handle * handle )
|
2008-04-05 20:12:11 +00:00
|
|
|
{
|
|
|
|
return handle->configDir;
|
2006-07-16 19:39:23 +00:00
|
|
|
}
|
|
|
|
|
2007-06-18 19:39:52 +00:00
|
|
|
const char *
|
2008-04-05 20:12:11 +00:00
|
|
|
tr_getTorrentDir( const tr_handle * handle )
|
2006-07-16 19:39:23 +00:00
|
|
|
{
|
2008-04-05 20:12:11 +00:00
|
|
|
return handle->torrentDir;
|
|
|
|
}
|
2006-07-16 19:39:23 +00:00
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
const char *
|
|
|
|
tr_getResumeDir( const tr_handle * handle )
|
|
|
|
{
|
|
|
|
return handle->resumeDir;
|
|
|
|
}
|
2007-06-18 19:39:52 +00:00
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
const char*
|
|
|
|
tr_getDefaultConfigDir( void )
|
|
|
|
{
|
|
|
|
static char * s = NULL;
|
2006-07-16 19:39:23 +00:00
|
|
|
|
2008-04-05 20:12:11 +00:00
|
|
|
if( !s )
|
|
|
|
{
|
2008-09-23 19:11:04 +00:00
|
|
|
if( ( s = getenv( "TRANSMISSION_HOME" ) ) )
|
2008-04-05 20:12:11 +00:00
|
|
|
{
|
2008-10-14 03:03:29 +00:00
|
|
|
s = tr_strdup( s );
|
2008-04-05 20:12:11 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-04-05 20:27:13 +00:00
|
|
|
#ifdef SYS_DARWIN
|
2008-10-14 03:03:29 +00:00
|
|
|
s = tr_buildPath( getHomeDir( ), "Library",
|
|
|
|
"Application Support", "Transmission", NULL );
|
2008-09-23 19:11:04 +00:00
|
|
|
#elif defined( WIN32 )
|
2008-10-20 19:34:19 +00:00
|
|
|
char appdata[MAX_PATH]; /* SHGetFolderPath() requires MAX_PATH */
|
|
|
|
SHGetFolderPath( NULL, CSIDL_APPDATA, NULL, 0, appdata );
|
|
|
|
s = tr_buildPath( appdata, "Transmission", NULL );
|
2006-07-16 19:39:23 +00:00
|
|
|
#else
|
2008-09-23 19:11:04 +00:00
|
|
|
if( ( s = getenv( "XDG_CONFIG_HOME" ) ) )
|
2008-10-14 03:03:29 +00:00
|
|
|
s = tr_buildPath( s, "transmission", NULL );
|
2008-04-05 20:12:11 +00:00
|
|
|
else
|
2008-10-14 03:03:29 +00:00
|
|
|
s = tr_buildPath( getHomeDir( ), ".config", "transmission", NULL );
|
2006-07-16 19:39:23 +00:00
|
|
|
#endif
|
2008-04-05 20:12:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return s;
|
2006-07-16 19:39:23 +00:00
|
|
|
}
|
2008-02-28 19:06:23 +00:00
|
|
|
|
2008-07-11 04:07:14 +00:00
|
|
|
/***
|
|
|
|
****
|
|
|
|
***/
|
|
|
|
|
|
|
|
static int
|
|
|
|
isClutchDir( const char * path )
|
|
|
|
{
|
|
|
|
struct stat sb;
|
2008-10-14 03:03:29 +00:00
|
|
|
char * tmp = tr_buildPath( path, "javascript", "transmission.js", NULL );
|
|
|
|
const int ret = !stat( tmp, &sb );
|
2008-07-11 17:09:53 +00:00
|
|
|
tr_inf( _( "Searching for web interface file \"%s\"" ), tmp );
|
2008-10-14 03:03:29 +00:00
|
|
|
tr_free( tmp );
|
|
|
|
return ret;
|
|
|
|
|
2008-07-11 04:07:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
tr_getClutchDir( const tr_session * session UNUSED )
|
|
|
|
{
|
|
|
|
static char * s = NULL;
|
|
|
|
|
|
|
|
if( !s )
|
|
|
|
{
|
2008-09-23 19:11:04 +00:00
|
|
|
if( ( s = getenv( "CLUTCH_HOME" ) ) )
|
2008-07-11 04:07:14 +00:00
|
|
|
{
|
2008-10-14 03:03:29 +00:00
|
|
|
s = tr_strdup( s );
|
2008-07-11 04:07:14 +00:00
|
|
|
}
|
2008-09-23 19:11:04 +00:00
|
|
|
else if( ( s = getenv( "TRANSMISSION_WEB_HOME" ) ) )
|
2008-07-14 16:00:20 +00:00
|
|
|
{
|
2008-10-14 03:03:29 +00:00
|
|
|
s = tr_strdup( s );
|
2008-07-14 16:00:20 +00:00
|
|
|
}
|
2008-07-11 04:07:14 +00:00
|
|
|
else
|
|
|
|
{
|
2008-10-19 17:43:04 +00:00
|
|
|
|
2008-07-11 04:07:14 +00:00
|
|
|
#ifdef SYS_DARWIN
|
2008-10-19 17:43:04 +00:00
|
|
|
|
|
|
|
CFURLRef appURL = CFBundleCopyBundleURL( CFBundleGetMainBundle( ) );
|
|
|
|
CFStringRef appRef = CFURLCopyFileSystemPath( appURL,
|
|
|
|
kCFURLPOSIXPathStyle );
|
|
|
|
const char * appString = CFStringGetCStringPtr( appRef,
|
|
|
|
CFStringGetFastestEncoding( appRef ) );
|
2008-08-22 23:55:33 +00:00
|
|
|
CFRelease( appURL );
|
|
|
|
CFRelease( appRef );
|
2008-07-11 04:19:23 +00:00
|
|
|
|
2008-10-14 03:03:29 +00:00
|
|
|
s = tr_buildPath( appString, "Contents", "Resources", "web", NULL );
|
2008-10-19 17:43:04 +00:00
|
|
|
|
2008-09-23 19:11:04 +00:00
|
|
|
#elif defined( WIN32 )
|
|
|
|
|
2008-10-19 17:43:04 +00:00
|
|
|
/* SHGetFolderPath explicitly requires MAX_PATH length */
|
|
|
|
char dir[MAX_PATH];
|
|
|
|
|
|
|
|
/* Generally, Web interface should be stored in a Web subdir of
|
|
|
|
* calling executable dir. */
|
|
|
|
|
|
|
|
if( s == NULL ) {
|
|
|
|
/* First, we should check personal AppData/Transmission/Web */
|
|
|
|
SHGetFolderPath( NULL, CSIDL_COMMON_APPDATA, NULL, 0, dir );
|
|
|
|
s = tr_buildPath( dir, "Transmission", "Web", NULL );
|
|
|
|
if( !isClutchDir( s ) ) {
|
|
|
|
tr_free( s );
|
|
|
|
s = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if( s == NULL ) {
|
|
|
|
/* check personal AppData */
|
|
|
|
SHGetFolderPath( NULL, CSIDL_APPDATA, NULL, 0, dir );
|
|
|
|
s = tr_buildPath( dir, "Transmission", "Web", NULL );
|
|
|
|
if( !isClutchDir( s ) ) {
|
|
|
|
tr_free( s );
|
|
|
|
s = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if( s == NULL) {
|
|
|
|
/* check calling module place */
|
|
|
|
GetModuleFileName( GetModuleHandle( NULL ), dir, sizeof( dir ) );
|
|
|
|
s = tr_buildPath( dirname( dir ), "Web", NULL );
|
|
|
|
if( !isClutchDir( s ) ) {
|
|
|
|
tr_free( s );
|
|
|
|
s = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#else /* everyone else, follow the XDG spec */
|
|
|
|
|
2008-09-23 19:11:04 +00:00
|
|
|
tr_list *candidates = NULL, *l;
|
2008-07-11 04:07:14 +00:00
|
|
|
|
|
|
|
/* XDG_DATA_HOME should be the first in the list of candidates */
|
|
|
|
s = getenv( "XDG_DATA_HOME" );
|
|
|
|
if( s && *s )
|
|
|
|
tr_list_append( &candidates, tr_strdup( s ) );
|
2008-10-14 03:03:29 +00:00
|
|
|
else {
|
|
|
|
char * dhome = tr_buildPath( getHomeDir( ), ".local", "share", NULL );
|
|
|
|
tr_list_append( &candidates, dhome );
|
2008-07-11 04:07:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* XDG_DATA_DIRS are the backup directories */
|
|
|
|
s = getenv( "XDG_DATA_DIRS" );
|
|
|
|
if( !s || !*s )
|
2008-08-14 10:35:11 +00:00
|
|
|
s = PACKAGE_DATA_DIR ":/usr/local/share/:/usr/share/";
|
2008-09-23 19:11:04 +00:00
|
|
|
while( s && *s )
|
|
|
|
{
|
2008-07-11 04:07:14 +00:00
|
|
|
char * end = strchr( s, ':' );
|
2008-09-23 19:11:04 +00:00
|
|
|
if( end )
|
|
|
|
{
|
|
|
|
tr_list_append( &candidates, tr_strndup( s, end - s ) );
|
2008-07-11 04:07:14 +00:00
|
|
|
s = end + 1;
|
2008-09-23 19:11:04 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2008-07-11 04:07:14 +00:00
|
|
|
tr_list_append( &candidates, tr_strdup( s ) );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-09-23 19:11:04 +00:00
|
|
|
for( l = candidates; l; l = l->next )
|
|
|
|
{
|
2008-10-14 03:03:29 +00:00
|
|
|
char * path = tr_buildPath( l->data, "transmission", "web", NULL );
|
|
|
|
const int found = isClutchDir( path );
|
|
|
|
if( found ) {
|
|
|
|
s = path;
|
2008-07-11 04:07:14 +00:00
|
|
|
break;
|
2008-10-14 03:03:29 +00:00
|
|
|
}
|
2008-10-14 17:27:59 +00:00
|
|
|
tr_free( path );
|
2008-07-11 04:07:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
tr_list_free( &candidates, tr_free );
|
2008-10-19 17:43:04 +00:00
|
|
|
|
2008-07-11 04:07:14 +00:00
|
|
|
#endif
|
2008-10-19 17:43:04 +00:00
|
|
|
|
2008-07-11 04:07:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2008-02-28 19:06:23 +00:00
|
|
|
/***
|
|
|
|
****
|
|
|
|
***/
|
|
|
|
|
2008-08-17 12:39:26 +00:00
|
|
|
tr_lockfile_state_t
|
2008-02-28 19:06:23 +00:00
|
|
|
tr_lockfile( const char * filename )
|
|
|
|
{
|
2008-08-17 12:39:26 +00:00
|
|
|
tr_lockfile_state_t ret;
|
2008-02-28 19:06:23 +00:00
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
|
2008-09-23 19:11:04 +00:00
|
|
|
HANDLE file = CreateFile(
|
|
|
|
filename,
|
|
|
|
GENERIC_READ | GENERIC_WRITE,
|
|
|
|
FILE_SHARE_READ |
|
|
|
|
FILE_SHARE_WRITE,
|
|
|
|
NULL,
|
|
|
|
OPEN_ALWAYS,
|
|
|
|
FILE_ATTRIBUTE_NORMAL,
|
|
|
|
NULL );
|
2008-02-28 19:06:23 +00:00
|
|
|
if( file == INVALID_HANDLE_VALUE )
|
|
|
|
ret = TR_LOCKFILE_EOPEN;
|
|
|
|
else if( !LockFile( file, 0, 0, 1, 1 ) )
|
|
|
|
ret = TR_LOCKFILE_ELOCK;
|
|
|
|
else
|
|
|
|
ret = TR_LOCKFILE_SUCCESS;
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
int fd = open( filename, O_RDWR | O_CREAT, 0666 );
|
|
|
|
if( fd < 0 )
|
|
|
|
ret = TR_LOCKFILE_EOPEN;
|
2008-09-23 19:11:04 +00:00
|
|
|
else
|
|
|
|
{
|
2008-02-28 19:06:23 +00:00
|
|
|
struct flock lk;
|
|
|
|
memset( &lk, 0, sizeof( lk ) );
|
|
|
|
lk.l_start = 0;
|
|
|
|
lk.l_len = 0;
|
|
|
|
lk.l_type = F_WRLCK;
|
|
|
|
lk.l_whence = SEEK_SET;
|
|
|
|
if( -1 == fcntl( fd, F_SETLK, &lk ) )
|
|
|
|
ret = TR_LOCKFILE_ELOCK;
|
|
|
|
else
|
|
|
|
ret = TR_LOCKFILE_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2008-09-05 14:31:52 +00:00
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
|
|
|
|
/* The following mmap functions are by Joerg Walter, and were taken from
|
|
|
|
* his paper at: http://www.genesys-e.de/jwalter/mix4win.htm
|
|
|
|
*/
|
|
|
|
|
2008-10-14 17:30:07 +00:00
|
|
|
#if defined(_MSC_VER)
|
|
|
|
__declspec( align( 4 ) ) static LONG volatile g_sl;
|
|
|
|
#else
|
2008-09-23 19:11:04 +00:00
|
|
|
static LONG volatile g_sl __attribute__ ( ( aligned ( 4 ) ) );
|
2008-10-14 17:30:07 +00:00
|
|
|
#endif
|
2008-09-05 14:32:28 +00:00
|
|
|
|
2008-09-05 14:31:52 +00:00
|
|
|
/* Wait for spin lock */
|
2008-09-23 19:11:04 +00:00
|
|
|
static int
|
|
|
|
slwait( LONG volatile *sl )
|
|
|
|
{
|
|
|
|
while( InterlockedCompareExchange ( sl, 1, 0 ) != 0 )
|
|
|
|
Sleep ( 0 );
|
|
|
|
|
2008-09-05 14:31:52 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Release spin lock */
|
2008-09-23 19:11:04 +00:00
|
|
|
static int
|
2008-10-14 17:30:07 +00:00
|
|
|
slrelease( LONG volatile *sl )
|
2008-09-23 19:11:04 +00:00
|
|
|
{
|
|
|
|
InterlockedExchange ( sl, 0 );
|
2008-09-05 14:31:52 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-09-05 14:32:32 +00:00
|
|
|
/* getpagesize for windows */
|
2008-09-23 19:11:04 +00:00
|
|
|
static long
|
|
|
|
getpagesize( void )
|
|
|
|
{
|
2008-09-05 14:32:32 +00:00
|
|
|
static long g_pagesize = 0;
|
2008-09-23 19:11:04 +00:00
|
|
|
|
|
|
|
if( !g_pagesize )
|
|
|
|
{
|
2008-09-05 14:32:32 +00:00
|
|
|
SYSTEM_INFO system_info;
|
2008-09-23 19:11:04 +00:00
|
|
|
GetSystemInfo ( &system_info );
|
2008-09-05 14:32:32 +00:00
|
|
|
g_pagesize = system_info.dwPageSize;
|
|
|
|
}
|
|
|
|
return g_pagesize;
|
|
|
|
}
|
|
|
|
|
2008-09-23 19:11:04 +00:00
|
|
|
static long
|
|
|
|
getregionsize( void )
|
|
|
|
{
|
2008-09-05 14:32:32 +00:00
|
|
|
static long g_regionsize = 0;
|
2008-09-23 19:11:04 +00:00
|
|
|
|
|
|
|
if( !g_regionsize )
|
|
|
|
{
|
2008-09-05 14:32:32 +00:00
|
|
|
SYSTEM_INFO system_info;
|
2008-09-23 19:11:04 +00:00
|
|
|
GetSystemInfo ( &system_info );
|
2008-09-05 14:32:32 +00:00
|
|
|
g_regionsize = system_info.dwAllocationGranularity;
|
|
|
|
}
|
|
|
|
return g_regionsize;
|
|
|
|
}
|
|
|
|
|
2008-09-23 19:11:04 +00:00
|
|
|
void *
|
|
|
|
mmap( void *ptr,
|
|
|
|
long size,
|
|
|
|
long prot,
|
|
|
|
long type,
|
|
|
|
long handle,
|
|
|
|
long arg )
|
|
|
|
{
|
2008-09-05 14:31:52 +00:00
|
|
|
static long g_pagesize;
|
|
|
|
static long g_regionsize;
|
2008-09-23 19:11:04 +00:00
|
|
|
|
2008-09-05 14:31:52 +00:00
|
|
|
/* Wait for spin lock */
|
2008-09-23 19:11:04 +00:00
|
|
|
slwait ( &g_sl );
|
2008-09-05 14:31:52 +00:00
|
|
|
/* First time initialization */
|
2008-09-23 19:11:04 +00:00
|
|
|
if( !g_pagesize )
|
|
|
|
g_pagesize = getpagesize ( );
|
|
|
|
if( !g_regionsize )
|
|
|
|
g_regionsize = getregionsize ( );
|
2008-09-05 14:31:52 +00:00
|
|
|
/* Allocate this */
|
2008-09-23 19:11:04 +00:00
|
|
|
ptr = VirtualAlloc ( ptr, size,
|
|
|
|
MEM_RESERVE | MEM_COMMIT | MEM_TOP_DOWN,
|
|
|
|
PAGE_READWRITE );
|
|
|
|
if( !ptr )
|
|
|
|
{
|
|
|
|
ptr = (void *) -1;
|
2008-09-05 14:31:52 +00:00
|
|
|
goto mmap_exit;
|
|
|
|
}
|
|
|
|
mmap_exit:
|
|
|
|
/* Release spin lock */
|
2008-09-23 19:11:04 +00:00
|
|
|
slrelease ( &g_sl );
|
2008-09-05 14:31:52 +00:00
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
2008-09-23 19:11:04 +00:00
|
|
|
long
|
|
|
|
munmap( void *ptr,
|
|
|
|
long size )
|
|
|
|
{
|
2008-09-05 14:31:52 +00:00
|
|
|
static long g_pagesize;
|
|
|
|
static long g_regionsize;
|
2008-09-23 19:11:04 +00:00
|
|
|
int rc = -1;
|
|
|
|
|
2008-09-05 14:31:52 +00:00
|
|
|
/* Wait for spin lock */
|
2008-09-23 19:11:04 +00:00
|
|
|
slwait ( &g_sl );
|
2008-09-05 14:31:52 +00:00
|
|
|
/* First time initialization */
|
2008-09-23 19:11:04 +00:00
|
|
|
if( !g_pagesize )
|
|
|
|
g_pagesize = getpagesize ( );
|
|
|
|
if( !g_regionsize )
|
|
|
|
g_regionsize = getregionsize ( );
|
2008-09-05 14:31:52 +00:00
|
|
|
/* Free this */
|
2008-09-23 19:11:04 +00:00
|
|
|
if( !VirtualFree ( ptr, 0,
|
|
|
|
MEM_RELEASE ) )
|
2008-09-05 14:31:52 +00:00
|
|
|
goto munmap_exit;
|
|
|
|
rc = 0;
|
|
|
|
munmap_exit:
|
|
|
|
/* Release spin lock */
|
2008-09-23 19:11:04 +00:00
|
|
|
slrelease ( &g_sl );
|
2008-09-05 14:31:52 +00:00
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|