/* Support for thread stuff. * * JC & KM 9/5/94 * Modified: * 28/11/94 JC * - return(0) missing from tidy_thread_info() * 4/8/99 RP JC * - reorganised for POSIX * 28/3/11 * - moved to vips_ namespace */ /* This file is part of VIPS. VIPS is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA */ /* These files are distributed with VIPS - http://www.vips.ecs.soton.ac.uk */ /* #define DEBUG_IO */ #ifdef HAVE_CONFIG_H #include #endif /*HAVE_CONFIG_H*/ #include #include #include #include #include #include #include void vips_semaphore_init( VipsSemaphore *s, int v, char *name ) { s->v = v; s->name = name; s->mutex = vips_g_mutex_new(); s->cond = vips_g_cond_new(); } void vips_semaphore_destroy( VipsSemaphore *s ) { VIPS_FREEF( vips_g_mutex_free, s->mutex ); VIPS_FREEF( vips_g_cond_free, s->cond ); } /* Add n to the semaphore and signal any threads that are blocked waiting * a change. */ int vips_semaphore_upn( VipsSemaphore *s, int n ) { int value_after_op; g_mutex_lock( s->mutex ); s->v += n; value_after_op = s->v; /* If we are only incrementing by one, we only need to wake a single * thread. If we are incrementing by a lot, we must wake all threads. */ if( n == 1 ) g_cond_signal( s->cond ); else g_cond_broadcast( s->cond ); g_mutex_unlock( s->mutex ); #ifdef DEBUG_IO printf( "vips_semaphore_upn(\"%s\",%d) = %d\n", s->name, n, value_after_op ); if( value_after_op > 1 ) vips_error( "vips_semaphore_upn", "up over 1!" ); #endif /*DEBUG_IO*/ return( value_after_op ); } /* Increment the semaphore. */ int vips_semaphore_up( VipsSemaphore *s ) { return( vips_semaphore_upn( s, 1 ) ); } /* Wait for sem>n, then subtract n. * Returns -1 when the monotonic time in @end_time was passed. */ static int vips__semaphore_downn_until( VipsSemaphore *s, int n, gint64 end_time ) { int value_after_op; VIPS_GATE_START( "vips__semaphore_downn_until: wait" ); g_mutex_lock( s->mutex ); while( s->v < n ) { if( end_time == -1 ) g_cond_wait( s->cond, s->mutex ); else if( !g_cond_wait_until( s->cond, s->mutex, end_time ) ) { /* timeout has passed. */ g_mutex_unlock( s->mutex ); VIPS_GATE_STOP( "vips__semaphore_downn_until: wait" ); return( -1 ); } } s->v -= n; value_after_op = s->v; g_mutex_unlock( s->mutex ); #ifdef DEBUG_IO printf( "vips__semaphore_downn_until(\"%s\",%d): %d\n", s->name, n, value_after_op ); #endif /*DEBUG_IO*/ VIPS_GATE_STOP( "vips__semaphore_downn_until: wait" ); return( value_after_op ); } /* Wait for sem>n, then subtract n. */ int vips_semaphore_downn( VipsSemaphore *s, int n ) { return( vips__semaphore_downn_until( s, n, -1 ) ); } /* Wait for sem > 0, then decrement. */ int vips_semaphore_down( VipsSemaphore *s ) { return( vips__semaphore_downn_until( s, 1, -1 ) ); } /* Wait for sem > 0, then decrement. * Returns -1 when the relative time in @timeout (in microseconds) * was passed. */ int vips_semaphore_down_timeout( VipsSemaphore *s, gint64 timeout ) { gint64 end_time = g_get_monotonic_time () + timeout; return( vips__semaphore_downn_until( s, 1, end_time ) ); }