source: trunk/libtransmission/platform.c @ 2544

Last change on this file since 2544 was 2544, checked in by charles, 14 years ago

this looks bug but it's not: just janitorial cleanup, moving #includes from headers into source file

  • Property svn:keywords set to Date Rev Author Id
File size: 19.1 KB
Line 
1/******************************************************************************
2 * $Id: platform.c 2544 2007-07-29 18:11:21Z charles $
3 *
4 * Copyright (c) 2005 Transmission authors and contributors
5 *
6 * Permission is hereby granted, free of charge, to any person obtaining a
7 * copy of this software and associated documentation files (the "Software"),
8 * to deal in the Software without restriction, including without limitation
9 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
10 * and/or sell copies of the Software, and to permit persons to whom the
11 * Software is furnished to do so, subject to the following conditions:
12 *
13 * The above copyright notice and this permission notice shall be included in
14 * all copies or substantial portions of the Software.
15 *
16 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
22 * DEALINGS IN THE SOFTWARE.
23 *****************************************************************************/
24
25#include <errno.h>
26#include <stdio.h>
27#include <stdlib.h>
28#include <string.h>
29
30#ifdef SYS_BEOS
31  #include <fs_info.h>
32  #include <FindDirectory.h>
33#endif
34#include <sys/types.h>
35#include <dirent.h>
36#include <fcntl.h>
37#include <unistd.h> /* getuid getpid close */
38
39#include "transmission.h"
40
41#if !defined( SYS_BEOS ) && !defined( __AMIGAOS4__ )
42
43#include <pwd.h>
44
45const char *
46tr_getHomeDirectory( void )
47{
48    static char     homeDirectory[MAX_PATH_LENGTH];
49    static int      init = 0;
50    char          * envHome;
51    struct passwd * pw;
52
53    if( init )
54    {
55        return homeDirectory;
56    }
57
58    envHome = getenv( "HOME" );
59    if( NULL == envHome )
60    {
61        pw = getpwuid( getuid() );
62        endpwent();
63        if( NULL == pw )
64        {
65            /* XXX need to handle this case */
66            return NULL;
67        }
68        envHome = pw->pw_dir;
69    }
70
71    snprintf( homeDirectory, MAX_PATH_LENGTH, "%s", envHome );
72    init = 1;
73
74    return homeDirectory;
75}
76
77#else
78
79const char *
80tr_getHomeDirectory( void )
81{
82    /* XXX */
83    return "";
84}
85
86#endif /* !SYS_BEOS && !__AMIGAOS4__ */
87
88static void
89tr_migrateResume( const char *oldDirectory, const char *newDirectory )
90{
91    DIR * dirh = opendir( oldDirectory );
92
93    if( dirh != NULL )
94    {
95        struct dirent * dirp;
96
97        while( ( dirp = readdir( dirh ) ) )
98        {
99            if( !strncmp( "resume.", dirp->d_name, 7 ) )
100            {
101                char o[MAX_PATH_LENGTH];
102                char n[MAX_PATH_LENGTH];
103                tr_buildPath( o, sizeof(o), oldDirectory, dirp->d_name, NULL );
104                tr_buildPath( n, sizeof(n), newDirectory, dirp->d_name, NULL );
105                rename( o, n );
106            }
107        }
108
109        closedir( dirh );
110    }
111}
112
113const char *
114tr_getPrefsDirectory( void )
115{
116    static char   buf[MAX_PATH_LENGTH];
117    static int    init = 0;
118    static size_t buflen = sizeof(buf);
119    const char* h;
120
121    if( init )
122        return buf;
123
124    h = tr_getHomeDirectory();
125#ifdef SYS_BEOS
126    find_directory( B_USER_SETTINGS_DIRECTORY,
127                    dev_for_path("/boot"), true, buf, buflen );
128    strcat( buf, "/Transmission" );
129#elif defined( SYS_DARWIN )
130    tr_buildPath ( buf, buflen, h,
131                  "Library", "Application Support", "Transmission", NULL );
132#elif defined(__AMIGAOS4__)
133    snprintf( buf, buflen, "PROGDIR:.transmission" );
134#else
135    tr_buildPath ( buf, buflen, h, ".transmission", NULL );
136#endif
137
138    tr_mkdir( buf );
139    init = 1;
140
141#ifdef SYS_DARWIN
142    char old[MAX_PATH_LENGTH];
143    tr_buildPath ( old, sizeof(old), h, ".transmission", NULL );
144    tr_migrateResume( old, buf );
145    rmdir( old );
146#endif
147
148    return buf;
149}
150
151const char *
152tr_getCacheDirectory( void )
153{
154    static char buf[MAX_PATH_LENGTH];
155    static int  init = 0;
156    static const size_t buflen = sizeof(buf);
157    const char * p;
158
159    if( init )
160        return buf;
161
162    p = tr_getPrefsDirectory();
163#ifdef SYS_BEOS
164    tr_buildPath( buf, buflen, p, "Cache", NULL );
165#elif defined( SYS_DARWIN )
166    tr_buildPath( buf, buflen, tr_getHomeDirectory(),
167                  "Library", "Caches", "Transmission", NULL );
168#else
169    tr_buildPath( buf, buflen, p, "cache", NULL );
170#endif
171
172    tr_mkdir( buf );
173    init = 1;
174
175    if( strcmp( p, buf ) )
176        tr_migrateResume( p, buf );
177
178    return buf;
179}
180
181const char *
182tr_getTorrentsDirectory( void )
183{
184    static char buf[MAX_PATH_LENGTH];
185    static int  init = 0;
186    static const size_t buflen = sizeof(buf);
187    const char * p;
188
189    if( init )
190        return buf;
191
192    p = tr_getPrefsDirectory ();
193
194#ifdef SYS_BEOS
195    tr_buildPath( buf, buflen, p, "Torrents", NULL );
196#elif defined( SYS_DARWIN )
197    tr_buildPath( buf, buflen, p, "Torrents", NULL );
198#else
199    tr_buildPath( buf, buflen, p, "torrents", NULL );
200#endif
201
202    tr_mkdir( buf );
203    init = 1;
204    return buf;
205}
206
207static void ThreadFunc( void * _t )
208{
209    tr_thread_t * t = _t;
210    char* name = tr_strdup( t->name );
211
212#ifdef SYS_BEOS
213    /* This is required because on BeOS, SIGINT is sent to each thread,
214       which kills them not nicely */
215    signal( SIGINT, SIG_IGN );
216#endif
217
218    tr_dbg( "Thread '%s' started", name );
219    t->func( t->arg );
220    tr_dbg( "Thread '%s' exited", name );
221    tr_free( name );
222}
223
224void tr_threadCreate( tr_thread_t * t,
225                      void (*func)(void *), void * arg,
226                      const char * name )
227{
228    t->func = func;
229    t->arg  = arg;
230    t->name = tr_strdup( name );
231#ifdef SYS_BEOS
232    t->thread = spawn_thread( (void *) ThreadFunc, name,
233                              B_NORMAL_PRIORITY, t );
234    resume_thread( t->thread );
235#else
236    pthread_create( &t->thread, NULL, (void * (*) (void *)) ThreadFunc, t );
237#endif
238}
239
240const tr_thread_t THREAD_EMPTY = { NULL, NULL, NULL, 0 };
241
242void tr_threadJoin( tr_thread_t * t )
243{
244    if( t->func != NULL )
245    {
246#ifdef SYS_BEOS
247        long exit;
248        wait_for_thread( t->thread, &exit );
249#else
250        pthread_join( t->thread, NULL );
251#endif
252        tr_dbg( "Thread '%s' joined", t->name );
253        tr_free( t->name );
254        t->name = NULL;
255        t->func = NULL;
256    }
257}
258
259void tr_lockInit( tr_lock_t * l )
260{
261#ifdef SYS_BEOS
262    *l = create_sem( 1, "" );
263#else
264    pthread_mutex_init( l, NULL );
265#endif
266}
267
268void tr_lockClose( tr_lock_t * l )
269{
270#ifdef SYS_BEOS
271    delete_sem( *l );
272#else
273    pthread_mutex_destroy( l );
274#endif
275}
276
277int tr_lockTryLock( tr_lock_t * l )
278{
279#ifdef SYS_BEOS
280    return acquire_sem_etc( *l, 1, B_RELATIVE_TIMEOUT, 0 );
281#else
282    /* success on zero! */
283    return pthread_mutex_trylock( l );
284#endif
285}
286
287void tr_lockLock( tr_lock_t * l )
288{
289#ifdef SYS_BEOS
290    acquire_sem( *l );
291#else
292    pthread_mutex_lock( l );
293#endif
294}
295
296void tr_lockUnlock( tr_lock_t * l )
297{
298#ifdef SYS_BEOS
299    release_sem( *l );
300#else
301    pthread_mutex_unlock( l );
302#endif
303}
304
305
306void tr_condInit( tr_cond_t * c )
307{
308#ifdef SYS_BEOS
309    c->sem = create_sem( 1, "" );
310    c->start = 0;
311    c->end = 0;
312#else
313    pthread_cond_init( c, NULL );
314#endif
315}
316
317void tr_condWait( tr_cond_t * c, tr_lock_t * l )
318{
319#ifdef SYS_BEOS
320    /* Keep track of that thread */
321    acquire_sem( c->sem );
322    c->threads[c->end] = find_thread( NULL );
323    c->end = ( c->end + 1 ) % BEOS_MAX_THREADS;
324    assert( c->end != c->start ); /* We hit BEOS_MAX_THREADS, arggh */
325    release_sem( c->sem );
326
327    release_sem( *l );
328    suspend_thread( find_thread( NULL ) ); /* Wait for signal */
329    acquire_sem( *l );
330#else
331    pthread_cond_wait( c, l );
332#endif
333}
334
335#ifdef SYS_BEOS
336static int condTrySignal( tr_cond_t * c )
337{
338    if( c->start == c->end )
339        return 1;
340
341    for( ;; )
342    {
343        thread_info info;
344        get_thread_info( c->threads[c->start], &info );
345        if( info.state == B_THREAD_SUSPENDED )
346        {
347            resume_thread( c->threads[c->start] );
348            c->start = ( c->start + 1 ) % BEOS_MAX_THREADS;
349            break;
350        }
351        /* The thread is not suspended yet, which can happen since
352         * tr_condWait does not atomically suspends after releasing
353         * the semaphore. Wait a bit and try again. */
354        snooze( 5000 );
355    }
356    return 0;
357}
358#endif
359void tr_condSignal( tr_cond_t * c )
360{
361#ifdef SYS_BEOS
362    acquire_sem( c->sem );
363    condTrySignal( c );
364    release_sem( c->sem );
365#else
366    pthread_cond_signal( c );
367#endif
368}
369void tr_condBroadcast( tr_cond_t * c )
370{
371#ifdef SYS_BEOS
372    acquire_sem( c->sem );
373    while( !condTrySignal( c ) );
374    release_sem( c->sem );
375#else
376    pthread_cond_broadcast( c );
377#endif
378}
379
380void tr_condClose( tr_cond_t * c )
381{
382#ifdef SYS_BEOS
383    delete_sem( c->sem );
384#else
385    pthread_cond_destroy( c );
386#endif
387}
388
389
390#if defined( BSD )
391
392#include <sys/types.h>
393#include <sys/socket.h>
394#include <netinet/in.h>
395#include <arpa/inet.h>
396#include <sys/sysctl.h>
397#include <net/route.h>
398
399static uint8_t *
400getroute( int * buflen );
401static int
402parseroutes( uint8_t * buf, int len, struct in_addr * addr );
403
404int
405tr_getDefaultRoute( struct in_addr * addr )
406{
407    uint8_t * buf;
408    int len;
409
410    buf = getroute( &len );
411    if( NULL == buf )
412    {
413        tr_err( "failed to get default route (BSD)" );
414        return 1;
415    }
416
417    len = parseroutes( buf, len, addr );
418    free( buf );
419
420    return len;
421}
422
423#ifndef SA_SIZE
424#define ROUNDUP( a, size ) \
425    ( ( (a) & ( (size) - 1 ) ) ? ( 1 + ( (a) | ( (size) - 1 ) ) ) : (a) )
426#define SA_SIZE( sap ) \
427    ( sap->sa_len ? ROUNDUP( (sap)->sa_len, sizeof( u_long ) ) : \
428                    sizeof( u_long ) )
429#endif /* !SA_SIZE */
430#define NEXT_SA( sap ) \
431    (struct sockaddr *) ( (caddr_t) (sap) + ( SA_SIZE( (sap) ) ) )
432
433static uint8_t *
434getroute( int * buflen )
435{
436    int     mib[6];
437    size_t  len;
438    uint8_t * buf;
439
440    mib[0] = CTL_NET;
441    mib[1] = PF_ROUTE;
442    mib[2] = 0;
443    mib[3] = AF_INET;
444    mib[4] = NET_RT_FLAGS;
445    mib[5] = RTF_GATEWAY;
446
447    if( sysctl( mib, 6, NULL, &len, NULL, 0 ) )
448    {
449        if( ENOENT != errno )
450        {
451            tr_err( "sysctl net.route.0.inet.flags.gateway failed (%s)",
452                    strerror( errno ) );
453        }
454        *buflen = 0;
455        return NULL;
456    }
457
458    buf = malloc( len );
459    if( NULL == buf )
460    {
461        *buflen = 0;
462        return NULL;
463    }
464
465    if( sysctl( mib, 6, buf, &len, NULL, 0 ) )
466    {
467        tr_err( "sysctl net.route.0.inet.flags.gateway failed (%s)",
468                strerror( errno ) );
469        free( buf );
470        *buflen = 0;
471        return NULL;
472    }
473
474    *buflen = len;
475
476    return buf;
477}
478
479static int
480parseroutes( uint8_t * buf, int len, struct in_addr * addr )
481{
482    uint8_t            * end;
483    struct rt_msghdr   * rtm;
484    struct sockaddr    * sa;
485    struct sockaddr_in * sin;
486    int                  ii;
487    struct in_addr       dest, gw;
488
489    end = buf + len;
490    while( end > buf + sizeof( *rtm ) )
491    {
492        rtm = (struct rt_msghdr *) buf;
493        buf += rtm->rtm_msglen;
494        if( end >= buf )
495        {
496            dest.s_addr = INADDR_NONE;
497            gw.s_addr   = INADDR_NONE;
498            sa = (struct sockaddr *) ( rtm + 1 );
499
500            for( ii = 0; ii < RTAX_MAX && (uint8_t *) sa < buf; ii++ )
501            {
502                if( buf < (uint8_t *) NEXT_SA( sa ) )
503                {
504                    break;
505                }
506
507                if( rtm->rtm_addrs & ( 1 << ii ) )
508                {
509                    if( AF_INET == sa->sa_family )
510                    {
511                        sin = (struct sockaddr_in *) sa;
512                        switch( ii )
513                        {
514                            case RTAX_DST:
515                                dest = sin->sin_addr;
516                                break;
517                            case RTAX_GATEWAY:
518                                gw = sin->sin_addr;
519                                break;
520                        }
521                    }
522                    sa = NEXT_SA( sa );
523                }
524            }
525
526            if( INADDR_ANY == dest.s_addr && INADDR_NONE != gw.s_addr )
527            {
528                *addr = gw;
529                return 0;
530            }
531        }
532    }
533
534    return 1;
535}
536
537#elif defined( linux ) || defined( __linux ) || defined( __linux__ )
538
539#include <linux/types.h>
540#include <linux/netlink.h>
541#include <linux/rtnetlink.h>
542
543#define SEQNUM 195909
544
545static int
546getsock( void );
547static uint8_t *
548getroute( int fd, unsigned int * buflen );
549static int
550parseroutes( uint8_t * buf, unsigned int len, struct in_addr * addr );
551
552int
553tr_getDefaultRoute( struct in_addr * addr )
554{
555    int fd, ret;
556    unsigned int len;
557    uint8_t * buf;
558
559    ret = 1;
560    fd = getsock();
561    if( 0 <= fd )
562    {
563        while( ret )
564        {
565            buf = getroute( fd, &len );
566            if( NULL == buf )
567            {
568                break;
569            }
570            ret = parseroutes( buf, len, addr );
571            free( buf );
572        }
573        close( fd );
574    }
575
576    if( ret )
577    {
578        tr_err( "failed to get default route (Linux)" );
579    }
580
581    return ret;
582}
583
584static int
585getsock( void )
586{
587    int fd, flags;
588    struct
589    {
590        struct nlmsghdr nlh;
591        struct rtgenmsg rtg;
592    } req;
593    struct sockaddr_nl snl;
594
595    fd = socket( PF_NETLINK, SOCK_DGRAM, NETLINK_ROUTE );
596    if( 0 > fd )
597    {
598        tr_err( "failed to create routing socket (%s)", strerror( errno ) );
599        return -1;
600    }
601
602    flags = fcntl( fd, F_GETFL );
603    if( 0 > flags || 0 > fcntl( fd, F_SETFL, O_NONBLOCK | flags ) )
604    {
605        tr_err( "failed to set socket nonblocking (%s)", strerror( errno ) );
606        close( fd );
607        return -1;
608    }
609
610    bzero( &snl, sizeof( snl ) );
611    snl.nl_family = AF_NETLINK;
612
613    bzero( &req, sizeof( req ) );
614    req.nlh.nlmsg_len = NLMSG_LENGTH( sizeof( req.rtg ) );
615    req.nlh.nlmsg_type = RTM_GETROUTE;
616    req.nlh.nlmsg_flags = NLM_F_REQUEST | NLM_F_DUMP;
617    req.nlh.nlmsg_seq = SEQNUM;
618    req.nlh.nlmsg_pid = 0;
619    req.rtg.rtgen_family = AF_INET;
620
621    if( 0 > sendto( fd, &req, sizeof( req ), 0,
622                    (struct sockaddr *) &snl, sizeof( snl ) ) )
623    {
624        tr_err( "failed to write to routing socket (%s)", strerror( errno ) );
625        close( fd );
626        return -1;
627    }
628
629    return fd;
630}
631
632static uint8_t *
633getroute( int fd, unsigned int * buflen )
634{
635    void             * buf;
636    unsigned int       len;
637    ssize_t            res;
638    struct sockaddr_nl snl;
639    socklen_t          slen;
640
641    len = 8192;
642    buf = calloc( 1, len );
643    if( NULL == buf )
644    {
645        *buflen = 0;
646        return NULL;
647    }
648
649    for( ;; )
650    {
651        bzero( &snl, sizeof( snl ) );
652        slen = sizeof( snl );
653        res = recvfrom( fd, buf, len, 0, (struct sockaddr *) &snl, &slen );
654        if( 0 > res )
655        {
656            if( EAGAIN != errno )
657            {
658                tr_err( "failed to read from routing socket (%s)",
659                        strerror( errno ) );
660            }
661            free( buf );
662            *buflen = 0;
663            return NULL;
664        }
665        if( slen < sizeof( snl ) || AF_NETLINK != snl.nl_family )
666        {
667            tr_err( "bad address" );
668            free( buf );
669            *buflen = 0;
670            return NULL;
671        }
672
673        if( 0 == snl.nl_pid )
674        {
675            break;
676        }
677    }
678
679    *buflen = res;
680
681    return buf;
682}
683
684static int
685parseroutes( uint8_t * buf, unsigned int len, struct in_addr * addr )
686{
687    struct nlmsghdr * nlm;
688    struct nlmsgerr * nle;
689    struct rtmsg    * rtm;
690    struct rtattr   * rta;
691    int               rtalen;
692    struct in_addr    gw, dst;
693
694    nlm = ( struct nlmsghdr * ) buf;
695    while( NLMSG_OK( nlm, len ) )
696    {
697        gw.s_addr = INADDR_ANY;
698        dst.s_addr = INADDR_ANY;
699        if( NLMSG_ERROR == nlm->nlmsg_type )
700        {
701            nle = (struct nlmsgerr *) NLMSG_DATA( nlm );
702            if( NLMSG_LENGTH( NLMSG_ALIGN( sizeof( struct nlmsgerr ) ) ) >
703                nlm->nlmsg_len )
704            {
705                tr_err( "truncated netlink error" );
706            }
707            else
708            {
709                tr_err( "netlink error (%s)", strerror( nle->error ) );
710            }
711            return 1;
712        }
713        else if( RTM_NEWROUTE == nlm->nlmsg_type && SEQNUM == nlm->nlmsg_seq &&
714                 getpid() == (pid_t) nlm->nlmsg_pid &&
715                 NLMSG_LENGTH( sizeof( struct rtmsg ) ) <= nlm->nlmsg_len )
716        {
717            rtm = NLMSG_DATA( nlm );
718            rta = RTM_RTA( rtm );
719            rtalen = RTM_PAYLOAD( nlm );
720
721            while( RTA_OK( rta, rtalen ) )
722            {
723                if( sizeof( struct in_addr ) <= RTA_PAYLOAD( rta ) )
724                {
725                    switch( rta->rta_type )
726                    {
727                        case RTA_GATEWAY:
728                            memcpy( &gw, RTA_DATA( rta ), sizeof( gw ) );
729                            break;
730                        case RTA_DST:
731                            memcpy( &dst, RTA_DATA( rta ), sizeof( dst ) );
732                            break;
733                    }
734                }
735                rta = RTA_NEXT( rta, rtalen );
736            }
737        }
738
739        if( INADDR_NONE != gw.s_addr && INADDR_ANY != gw.s_addr &&
740            INADDR_ANY == dst.s_addr )
741        {
742            *addr = gw;
743            return 0;
744        }
745
746        nlm = NLMSG_NEXT( nlm, len );
747    }
748
749    return 1;
750}
751
752#else /* not BSD or Linux */
753
754int
755tr_getDefaultRoute( struct in_addr * addr UNUSED )
756{
757    tr_inf( "don't know how to get default route on this platform" );
758    return 1;
759}
760
761#endif
762
763/***
764****
765***/
766
767static void
768tr_rwSignal( tr_rwlock_t * rw )
769{
770  if ( rw->wantToWrite )
771    tr_condSignal( &rw->writeCond );
772  else if ( rw->wantToRead )
773    tr_condBroadcast( &rw->readCond );
774}
775
776void
777tr_rwInit ( tr_rwlock_t * rw )
778{
779    memset( rw, 0, sizeof(tr_rwlock_t) );
780    tr_lockInit( &rw->lock );
781    tr_condInit( &rw->readCond );
782    tr_condInit( &rw->writeCond );
783}
784
785void
786tr_rwReaderLock( tr_rwlock_t * rw )
787{
788    tr_lockLock( &rw->lock );
789    rw->wantToRead++;
790    while( rw->haveWriter || rw->wantToWrite )
791        tr_condWait( &rw->readCond, &rw->lock );
792    rw->wantToRead--;
793    rw->readCount++;
794    tr_lockUnlock( &rw->lock );
795}
796
797int
798tr_rwReaderTrylock( tr_rwlock_t * rw )
799{
800    int ret = FALSE;
801    tr_lockLock( &rw->lock );
802    if ( !rw->haveWriter && !rw->wantToWrite ) {
803        rw->readCount++;
804        ret = TRUE;
805    }
806    tr_lockUnlock( &rw->lock );
807    return ret;
808
809}
810
811void
812tr_rwReaderUnlock( tr_rwlock_t * rw )
813{
814    tr_lockLock( &rw->lock );
815    --rw->readCount;
816    if( !rw->readCount )
817        tr_rwSignal( rw );
818    tr_lockUnlock( &rw->lock );
819}
820
821void
822tr_rwWriterLock( tr_rwlock_t * rw )
823{
824    tr_lockLock( &rw->lock );
825    rw->wantToWrite++;
826    while( rw->haveWriter || rw->readCount )
827        tr_condWait( &rw->writeCond, &rw->lock );
828    rw->wantToWrite--;
829    rw->haveWriter = TRUE;
830    tr_lockUnlock( &rw->lock );
831}
832
833int
834tr_rwWriterTrylock( tr_rwlock_t * rw )
835{
836    int ret = FALSE;
837    tr_lockLock( &rw->lock );
838    if( !rw->haveWriter && !rw->readCount )
839        ret = rw->haveWriter = TRUE;
840    tr_lockUnlock( &rw->lock );
841    return ret;
842}
843void
844tr_rwWriterUnlock( tr_rwlock_t * rw )
845{
846    tr_lockLock( &rw->lock );
847    rw->haveWriter = FALSE;
848    tr_rwSignal( rw );
849    tr_lockUnlock( &rw->lock );
850}
851
852void
853tr_rwClose( tr_rwlock_t * rw )
854{
855    tr_condClose( &rw->writeCond );
856    tr_condClose( &rw->readCond );
857    tr_lockClose( &rw->lock );
858}
Note: See TracBrowser for help on using the repository browser.